(转)MVC3+EF4.1学习系列(十)----MVC+EF处理树形结构
2011-11-02 18:07
369 查看
原文地址:http://www.cnblogs.com/wlflovenet/archive/2011/09/05/efandmvc11.html
文章索引和简介
通过前几篇文章 我们处理了 一对一, 一对多,多对多关系 很好的发挥了ORM框架的做用 但是 少说了一种 树形结构的处理, 而这种树形关系
我们也经常遇到,常见的N级类别的处理, 以及经常有数据与类别挂钩。今天主要写下EF处理树形结构以及 MVC如何展示树形结构。 前面几篇的例子
一直用的是一个例子,内容是连贯的。这篇是完全单独的~
先来说下工作中会遇到的常见场景 针对这几个场景来处理~
[b]1.类别[/b]
a.类别可以有无限级别
b.类别的最末端 不确定是第几级 某个节点 可以到二级 其他的节点 有可能到四级
c.tree型展示整个类别 并可以对tree进行CRUD (可以一次递归全部加载 也可以异步加载 )
d.面包屑型展示类别
e.删除父类 应把下面所有的子类删除
[b]2.与类别挂钩的数据 (本文是文章)[/b]
a. 可以根据任意级别的类别 查看文章
b. 合并两个类别的文章
上面这些场景 基本覆盖了类别操作的常见情况 如果大家觉得还有什么要处理 可以给我说 我补充上去~~
下面开始讲解~
一.准备工作
1.如何建立类别实体类 来展示树形结构
上代码
这样的设计 很好的展示了树形结构 一个节点有一个父类 多个子类 一个类别可以有多个文章 这里说下 后面的四个属性 不是必要的~
2.文章实体类
上代码 这个比较好理解 不解释了~
部分视图的视图
2.展示面包屑
树形结构的展示 我们经常遇到treeview型的 还会遇到另一种 面包屑这样的 如
我们这里要做的就是 根据当前类别 向上一层层推到最上面 并把最后一个加粗显示~ 我的思路是这样的 比如当前所在的类别 为最小的 广州 根据这个
一层层推到最上面 通过递归得到 广州>中国>亚洲 在通过反转字符串 并给加粗就行了~~ 代码如下
3.删除父类要把下面的子类全部删除
这里就涉及到一个树形结构的重要方法 通过当前类 得到该类的所有子类子子类等的ID集合 这时删除时用 delete in(子类集合 ) 就行了 ~~ 忘了
怎么直接执行SQL语句的~~ 去看上一篇文章.. 通过现在类 获得下面子类集合的方法 如下
依然是通过递归 获得所有的 子节点集合 再通过string.Join(",", cidList) 得到 delete in ( )
里需要的的 就行了~~
三.与类别挂钩的数据的展示
一. 可以根据任意级别的类别查看文章
依然提供两个方法~
1.根据类别 查询文章 利用递归
不过这个效率就太纠结了~~
2. 利用前面说到的 通过当前类 得到该类的所有子类子子类等的ID集合 再查询的时候 IN 就行了
这里有个小知识~ 用linq 执行 sql in的操作~ 用 Contains就行了 代码如下
二.合并两个类别的文章
这个很简单啦~ 合并两个类 就是把一个类下的文章id 都变成另一个 也就是说批量操作 不要用EF 的一个个更新就行
太慢了~还要发送多条更新语句
用context.Database.SqlQuery 直接执行 update 这个就不写代码啦~
四.通过第三方工具Telerik更加酷炫的展示tree
我上面写的东西 都是小打小闹 自娱自乐的玩下 有很多地方不完善 不合理,而MVC实现Tree 第三方工具已经有了帮我们做的非常优秀的了
这里给大家推荐个开源的 是Telerik的
tree------介绍与连接
把该有的操作基本全部都封装在了里面啦~非常方便 而且里面还有很多其他的控件~
还有,推荐大家看看源码 第三方工具一定要多看看实现 不要只会用
五.总结
这篇是完全独立的 和前面几篇没什么关系~ 代码贴的都是核心片段~ 聪明的大家看看就能明白了 而且应该有更好的实现
希望大家分享下EF MVC处理 tree的经验以及遇到的问题~~
我也提个问题 因为tree结构的操作 很多都用到了递归 在EF 高并发 大数据量处理时
我觉得会出现一些问题 希望大家说说怎么解决
总体内容没有太多难度~ 大家自己敲敲练练吧
实在太懒的同学 留个邮件 我把demo发给你们~
(ps:文章写在自己的22岁生日, 学习.net3年多了 希望自己通过不断的学习 再次进步~ 也希望在园子里 得到更多的人的指点与帮助 祝福自己! )
文章索引和简介
通过前几篇文章 我们处理了 一对一, 一对多,多对多关系 很好的发挥了ORM框架的做用 但是 少说了一种 树形结构的处理, 而这种树形关系
我们也经常遇到,常见的N级类别的处理, 以及经常有数据与类别挂钩。今天主要写下EF处理树形结构以及 MVC如何展示树形结构。 前面几篇的例子
一直用的是一个例子,内容是连贯的。这篇是完全单独的~
先来说下工作中会遇到的常见场景 针对这几个场景来处理~
[b]1.类别[/b]
a.类别可以有无限级别
b.类别的最末端 不确定是第几级 某个节点 可以到二级 其他的节点 有可能到四级
c.tree型展示整个类别 并可以对tree进行CRUD (可以一次递归全部加载 也可以异步加载 )
d.面包屑型展示类别
e.删除父类 应把下面所有的子类删除
[b]2.与类别挂钩的数据 (本文是文章)[/b]
a. 可以根据任意级别的类别 查看文章
b. 合并两个类别的文章
上面这些场景 基本覆盖了类别操作的常见情况 如果大家觉得还有什么要处理 可以给我说 我补充上去~~
下面开始讲解~
一.准备工作
1.如何建立类别实体类 来展示树形结构
上代码
/// <summary> /// 类别 /// </summary> public class Category { /// <summary> /// 主键 /// </summary> public int CategoryId { get; set; } /// <summary> /// 类别名字 /// </summary> [Required()] [StringLength(5)] public string CategoryName { get; set; } /// <summary> /// 父ID /// </summary> public Nullable<int> ParentId { get; set; } /// <summary> /// 上面的父节点 /// </summary> public virtual Category Parent { get; set; } /// <summary> /// 下面的子节点 /// </summary> [ForeignKey("ParentId")] public virtual ICollection<Category> ChildKeys { get; set; } /// <summary> /// 该类别的文章集合 /// </summary> public virtual ICollection<Article> articleList { get; set; } /// <summary> /// 编号 /// </summary> public string Note { get; set; } /// <summary> /// 状态 /// </summary> public string State { get; set; } /// <summary> /// 级别 /// </summary> public Nullable<int> Lev { get; set; } /// <summary> /// 排序 /// </summary> public int Sort { get; set; } }
这样的设计 很好的展示了树形结构 一个节点有一个父类 多个子类 一个类别可以有多个文章 这里说下 后面的四个属性 不是必要的~
2.文章实体类
上代码 这个比较好理解 不解释了~
部分视图的视图
@model TreeDemoCore.Model.Category <ul> @foreach (var childitem in Model.ChildKeys) { <li id=@childitem.CategoryId> @if (childitem.ChildKeys.Count > 0) { <img src="http://www.cnblogs.com/Content/img/noselectNode.jpg" /> } <span class="name">@childitem.CategoryName </span> @Html.ActionLink("添加", "Create") @Html.ActionLink("修改", "Edit") </li> } </ul>
2.展示面包屑
树形结构的展示 我们经常遇到treeview型的 还会遇到另一种 面包屑这样的 如
我们这里要做的就是 根据当前类别 向上一层层推到最上面 并把最后一个加粗显示~ 我的思路是这样的 比如当前所在的类别 为最小的 广州 根据这个
一层层推到最上面 通过递归得到 广州>中国>亚洲 在通过反转字符串 并给加粗就行了~~ 代码如下
/// <summary> /// 根据当前类别建造面包屑 /// </summary> /// <returns></returns> public static MvcHtmlString Menu(this HtmlHelper html, Category treeModel) { return new MvcHtmlString(MenuReverse(BindMenu(treeModel))); } /// <summary> /// 递归调用 得到 广州>中国>亚洲 /// </summary> /// <returns></returns> private static string BindMenu(Category Model) { StringBuilder sb = new StringBuilder(); sb.Append(Model.CategoryName); if (Model.Parent != null) { sb.Append(">"); sb.Append(BindMenu(Model.Parent)); } return sb.ToString(); } /// <summary> /// 反转字符串 并给最后一个加上黑体字标签 /// </summary> /// <returns></returns> private static string MenuReverse(string menu) { return string.Join(">", menu.Split('>').Select((s, i) => i == 0 ? string.Format("<strong>{0}</Strong>", s) : s).Reverse().ToArray()); }
3.删除父类要把下面的子类全部删除
这里就涉及到一个树形结构的重要方法 通过当前类 得到该类的所有子类子子类等的ID集合 这时删除时用 delete in(子类集合 ) 就行了 ~~ 忘了
怎么直接执行SQL语句的~~ 去看上一篇文章.. 通过现在类 获得下面子类集合的方法 如下
/// <summary> /// 获得父类下所有子类的集合 /// </summary> /// <returns></returns> private List<int> GetCidbyPid(int pid) { List<int> cidList = new List<int>(); Category CategoryModel = unitOfWork.CategoryRepository.GetTEntityByID(pid); foreach (var item in CategoryModel.ChildKeys) { cidList.Add(item.CategoryId); } foreach (var item in CategoryModel.ChildKeys) { cidList.AddRange(GetCidbyPid(item.CategoryId)); } return cidList; // 获得 1,2,3,4,5 tring strcid= string.Join(",", cidList); }
依然是通过递归 获得所有的 子节点集合 再通过string.Join(",", cidList) 得到 delete in ( )
里需要的的 就行了~~
三.与类别挂钩的数据的展示
一. 可以根据任意级别的类别查看文章
依然提供两个方法~
1.根据类别 查询文章 利用递归
不过这个效率就太纠结了~~
/// <summary> /// 通过任意类别 获得下面的全部文章 /// </summary> /// <param name="cid"></param> /// <returns></returns> private List<Article> GetArticleByCid(int cid) { List<Article> ArticleList=new List<Article>(); Category CategoryModel=unitOfWork.CategoryRepository.GetTEntityByID(cid); if (CategoryModel.ChildKeys.Count == 0) { ArticleList.AddRange(CategoryModel.articleList.ToList()); } foreach (var item in CategoryModel.ChildKeys) { ArticleList.AddRange(GetArticleByCid(item.CategoryId)); } return ArticleList; }
2. 利用前面说到的 通过当前类 得到该类的所有子类子子类等的ID集合 再查询的时候 IN 就行了
这里有个小知识~ 用linq 执行 sql in的操作~ 用 Contains就行了 代码如下
public ActionResult Index() { //准备测试数据 测试不同情况 GetCidbyPid为根据ID获得所有子类以及子子类等的集合 //测试最高级 List<int> CidList = GetCidbyPid(1); // var CidList = GetCidbyPid(2); //测试最低级 //var CidList = GetCidbyPid(6); // 贪婪加载类别 为了显示类别名字~ var ArticleList = unitOfWork.ArticleRepository.Get(c => CidList.Contains(c.CategoryId), includeProperties: "Category"); return View(ArticleList); }
二.合并两个类别的文章
这个很简单啦~ 合并两个类 就是把一个类下的文章id 都变成另一个 也就是说批量操作 不要用EF 的一个个更新就行
太慢了~还要发送多条更新语句
用context.Database.SqlQuery 直接执行 update 这个就不写代码啦~
四.通过第三方工具Telerik更加酷炫的展示tree
我上面写的东西 都是小打小闹 自娱自乐的玩下 有很多地方不完善 不合理,而MVC实现Tree 第三方工具已经有了帮我们做的非常优秀的了
这里给大家推荐个开源的 是Telerik的
tree------介绍与连接
把该有的操作基本全部都封装在了里面啦~非常方便 而且里面还有很多其他的控件~
还有,推荐大家看看源码 第三方工具一定要多看看实现 不要只会用
五.总结
这篇是完全独立的 和前面几篇没什么关系~ 代码贴的都是核心片段~ 聪明的大家看看就能明白了 而且应该有更好的实现
希望大家分享下EF MVC处理 tree的经验以及遇到的问题~~
我也提个问题 因为tree结构的操作 很多都用到了递归 在EF 高并发 大数据量处理时
我觉得会出现一些问题 希望大家说说怎么解决
总体内容没有太多难度~ 大家自己敲敲练练吧
实在太懒的同学 留个邮件 我把demo发给你们~
(ps:文章写在自己的22岁生日, 学习.net3年多了 希望自己通过不断的学习 再次进步~ 也希望在园子里 得到更多的人的指点与帮助 祝福自己! )
相关文章推荐
- MVC3+EF4.1学习系列(十)----MVC+EF处理树形结构
- MVC3+EF4.1学习系列(十)----MVC+EF处理树形结构
- MVC3+EF4.1学习系列(七)-----EF并发的处理
- (转)MVC3+EF4.1学习系列(十一)----EF4.1常见的问题解决
- MVC3+EF4.1学习系列(十一)----EF4.1常见的问题解决
- MVC3+EF4.1学习系列(九)-----EF4.1其他的一些技巧的使用
- MVC+EF处理树形结构
- MVC3+EF4.1学习系列(四)----- ORM关系的处理
- 关于WCF+MVC+EF 学习系列文章整理
- MVC3+EF4.1学习系列(十一)----EF4.1常见的问题解决(转)
- MVC3+EF4.1学习系列(九)-----EF4.1其他的一些技巧的使用
- MVC3+EF4.1学习系列
- MVC3+EF4.1学习系列(十一)----EF4.1常见的问题解决
- MVC系列学习(二)-初步了解ORM框架-EF
- ASP.NET MVC学习---(二)EF文件结构
- EXTJS学习系列提高篇:第二十九篇(转载)作者殷良胜,ext2.2打造Ext.form.ComboBox系列--树形结构
- ASP.NET MVC学习---(二)EF文件结构
- MVC3+EF4.1学习系列(十一)----EF4.1常见的问题解决
- MVC3+EF4.1学习系列(五)----- EF查找导航属性的几种方式
- MVC系列学习(三)-EF的延迟加载