本系列将使用zTree来创建、编辑关于品牌、车系、车型的无限级分类,使用datagrid显示,源码在github。先上最终效果:

datagrid显示所有记录、分页,提供添加、修改、删除按钮,并提供简单搜索:

创建分类,弹出模态窗口,zTree显示所有分类,点击勾选按钮或节点,所选节点名称显示到文本框:

提供客户端和服务端验证,验证不通过显示出错信息:

修改分类,弹出模态窗口,zTree显示当前选中的节点名称:

创建数据模型并生成到数据库

→创建CarModel.edmx,创建模型,无限级分类的一切"神奇"从ParentID字段开始。

→右键界面,选择"根据模型生成数据库"

→配置数据库连接,运行sql语句......等等,最终生成如下文件:

● App.config中的连接字符串需要复制到MVC主站点的web.config中。
● CarModel.Context.cs中包含了继承DbContext的上下文。
● CarModel.tt下包含了所有的领域模型Domain Models。
● CarModel.edmx.sql每次映射完执行里面的sql语句,将把数据同步到数据库。

架构设计

由于主要是体验无限级分类的增删改查,架构做得相对简单一些。其它组件列举的只是示意,实际只用到了缓存和序列化json的帮助类。如下:

□ IBaseRepository是接口的基类,提供了所有接口的泛型实现

using System; 
using System.Collections.Generic; 
using System.Linq.Expressions;
 
namespace Car.Test.DAL 
{ 
    public interface IBaseRepository<T> where T : class,new() 
    { 
        IEnumerable<T> LoadEntities(Expression<Func<T, bool>> whereLambda); 
        IEnumerable<T> LoadEntitiesByCache(Expression<Func<T, bool>> whereLambda); 
        T AddEntity(T entity); 
        T UpdateEntity(T entity); 
        void ClearCache(); 
        int SaveChanges(); 
    } 
}

□ 其它接口只需要继承该基类接口就可以

namespace Car.Test.DAL 
{ 
    public interface ICarCategoryRepository : IBaseRepository<Car.Test.Model.CarCategory> 
    { 
         
    } 
}

当然,如果Domain Model很多的话,这样写方便tt模版自动生成。

□ BaseRepository是Repository的基类,提供了所有Repository的泛型实现

提供了针对泛型的增删改查,还包括缓存查询,提交变化。

using System.Data; 
using System.Linq; 
using Car.Test.Cache; 
using Car.Test.Model; 
using System; 
using System.Collections.Generic; 
using System.Linq.Expressions;
 
namespace Car.Test.DAL 
{ 
    public class BaseRepository<T> where T:class,new() 
    { 
        protected CarModelContainer DataContext { get; private set; } 
        public ICacheProvider Cache { get; set; }
 
        public BaseRepository(ICacheProvider cacheProvider) 
        { 
            this.DataContext = new CarModelContainer(); 
            this.Cache = cacheProvider; 
        }
 
        public BaseRepository():this(new DefaultCacheProvider()){}
 
        public virtual IEnumerable<T> LoadEntities(Expression<Func<T, bool>> whereLambda) 
        { 
            IEnumerable<T> temp = DataContext.Set<T>().Where(whereLambda).AsEnumerable(); 
            return temp; 
        }
 
        public virtual IEnumerable<T> LoadEntitiesByCache(Expression<Func<T, bool>> whereLambda) 
        { 
            IEnumerable<T> entities = Cache.Get(typeof(T).ToString()) as IEnumerable<T>; 
            if (entities == null) 
            { 
                entities = DataContext.Set<T>().Where(whereLambda).AsEnumerable(); 
                if (entities.Any()) 
                { 
                    Cache.Set(typeof(T).ToString(),entities,3); 
                } 
            } 
            return entities; 
        }
 
        public virtual T AddEntity(T entity) 
        { 
            DataContext.Set<T>().Add(entity); 
            return entity; 
        }
 
        public virtual T UpdateEntity(T entity) 
        { 
            if (entity != null) 
            { 
                DataContext.Set<T>().Attach(entity); 
                DataContext.Entry(entity).State = EntityState.Modified;
 
            } 
            return entity; 
        }
 
        public void ClearCache() 
        { 
            Cache.InValidate(typeof(T).ToString()); 
        }
 
        public int SaveChanges() 
        { 
            return DataContext.SaveChanges(); 
        } 
    } 
}
 

□ 其它Repository只需继承BaseRepository并实现各自的接口

namespace Car.Test.DAL 
{ 
    public class CarCategoryRepository : BaseRepository<Car.Test.Model.CarCategory>,ICarCategoryRepository 
    { 
         
    } 
}

□ 缓存接口

namespace Car.Test.Cache 
{ 
    public interface ICacheProvider 
    { 
        object Get(string key); 
        void Set(string key, object data, int cacheTime); 
        bool IsSet(string key); 
        void InValidate(string key); 
    } 
}

□ 缓存实现,需要引入System.Runtime.Caching

using System; 
using System.Runtime.Caching;
 
namespace Car.Test.Cache 
{ 
    public class DefaultCacheProvider : ICacheProvider 
    { 
        private ObjectCache Cache 
        { 
            get { return MemoryCache.Default; } 
        }
 
        public object Get(string key) 
        { 
            return Cache[key]; 
        }
 
        public void Set(string key, object data, int cacheTime) 
        { 
            CacheItemPolicy policy = new CacheItemPolicy(); 
            policy.AbsoluteExpiration = DateTime.Now + TimeSpan.FromMinutes(cacheTime); 
            Cache.Add(new CacheItem(key, data), policy); 
        }
 
        public bool IsSet(string key) 
        { 
            return Cache[key] != null; 
        }
 
        public void InValidate(string key) 
        { 
            Cache.Remove(key); 
        } 
    } 
}
 

使用AutoMapper映射领域模型和视图模型

□ 视图模型

using DataAnnotationsExtensions; 
using System.ComponentModel.DataAnnotations;
 
namespace Car.Test.Portal.Models 
{ 
    public class CarCategoryVm 
    { 
        public int ID { get; set; }
 
        [Display(Name = "类名")] 
        [Required(ErrorMessage = "必填")] 
        [StringLength(10, MinimumLength = 2,ErrorMessage = "长度2-10位")] 
        public string Name { get; set; }
 
        [Display(Name = "前缀字母")] 
        [Required(ErrorMessage = "必填")] 
        [StringLength(1,ErrorMessage = "长度1位")] 
        public string PreLetter { get; set; }
 
        [Display(Name = "所属父级")] 
        [Required(ErrorMessage = "必填")] 
        public int ParentID { get; set; }
 
        public System.DateTime SubTime { get; set; }
 
        [Display(Name = "层级(根节点为0级)")] 
        [Required(ErrorMessage = "必填")] 
        [Min(1, ErrorMessage = "至少为1")] 
        public int Level { get; set; }
 
        [Display(Name = "是否为页节点")] 
        [Required(ErrorMessage = "必填")] 
        public bool IsLeaf { get; set; }
 
        public short Status { get; set; } 
        public short DelFlag { get; set; } 
    } 
}
 

引入DataAnnotationsExtensions组件,通过它可以设置最小值,关于DataAnnotationsExtensions的使用,在这里

□ 继承AutoMapper的Profile类,创建领域模型→视图模型映射

using AutoMapper; 
using Car.Test.Model; 
using Car.Test.Portal.Models;
 
namespace Car.Test.Portal.Helpers.AutoMapper 
{ 
    public class DomainToVmProfile : Profile 
    { 
        protected override void Configure() 
        { 
            Mapper.CreateMap<CarCategory, CarCategoryVm>(); 
        } 
    } 
}
 

□ 继承AutoMapper的Profile类,创建视图模型→领域模型映射

using AutoMapper; 
using Car.Test.Model; 
using Car.Test.Portal.Models;
 
namespace Car.Test.Portal.Helpers.AutoMapper 
{ 
    public class VmToDomainProfile : Profile 
    { 
        protected override void Configure() 
        { 
            Mapper.CreateMap<CarCategoryVm, CarCategory>() 
                .ForMember("Car", opt => opt.Ignore()); 
        } 
    } 
}
 

□ 初始化所有的Profile

using AutoMapper;
 
namespace Car.Test.Portal.Helpers.AutoMapper 
{ 
    public static class AutoMapperConfiguration 
    { 
        public static void Configure() 
        { 
            Mapper.Initialize(x => x.AddProfile<VmToDomainProfile>()); 
            Mapper.Initialize(x => x.AddProfile<DomainToVmProfile>()); 
        } 
    } 
}
 

□ 全局注册

        protected void Application_Start() 
        { 
            AreaRegistration.RegisterAllAreas();
 
            WebApiConfig.Register(GlobalConfiguration.Configuration); 
            FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); 
            RouteConfig.RegisterRoutes(RouteTable.Routes); 
            BundleConfig.RegisterBundles(BundleTable.Bundles);
 
            //全局配置映射 
            AutoMapperConfiguration.Configure(); 
        }

关于AutoMapper的使用,在这里

下篇进入无限级分类的增删改查环节。

MVC无限级分类01,分层架构,引入缓存,完成领域模型与视图模型的映射的更多相关文章

  1. MVC无限级分类02,增删改查

    继上一篇"MVC无限级分类01,分层架构,引入缓存,完成领域模型与视图模型的映射",本篇开始MVC无限级分类的增删改查部分,源码在github. 显示和查询 使用datagrid显 ...

  2. 说一说Web开发中两种常用的分层架构及其对应的代码模型

    昨天妹子让我帮她解决个问题,本以为可以轻松搞定,但是打开他们项目的一瞬间,我头皮发麻.本身功能不多的一个小项目,解决方案里竟然有几十个类库.仅仅搞明白各个类库的作用,代码层次之间的引用关系就花了一个多 ...

  3. 从Microsoft.AspNet.Identity看微软推荐的一种MVC的分层架构

    Microsoft.AspNet.Identity简介 Microsoft.AspNet.Identity是微软在MVC 5.0中新引入的一种membership框架,和之前ASP.NET传统的mem ...

  4. 移动App设计之分层架构+MVC

    http://www.cnblogs.com/Logen/archive/2012/11/08/2760638.html 场景分析:我们知道,一个移动设备的应用大多与网络有关,也就是说,我在移动设备上 ...

  5. 无限级分类Asp.net Mvc实现

    无限级分类Asp.net Mvc实现   无限级分类涉及到异步加载子类.加载当前类和匹配问题,现在做一个通用的实现.   (一) 效果如下:   (二)设计.实现及使用 (1)数据库 (a)表设计db ...

  6. 左右值无限级分类 MVC + EntityFramework 的简单实现

    在度娘上查了大半个月的资料,最后发现每个网友分享的文章都有一定的错误(PS:大家是故意的么?).最后是在看了一个ASP版本后知道了大概流程:看了一个存储过程实现的文章后知道了大概需要的功能:看了一个S ...

  7. DDD分层架构之领域实体(验证篇)

    DDD分层架构之领域实体(验证篇) 在应用程序框架实战十四:DDD分层架构之领域实体(基础篇)一文中,我介绍了领域实体的基础,包括标识.相等性比较.输出实体状态等.本文将介绍领域实体的一个核心内容—— ...

  8. 【转载】DDD分层架构的三种模式

    引言 在讨论DDD分层架构的模式之前,我们先一起回顾一下DDD和分层架构的相关知识. DDD DDD(Domain Driven Design,领域驱动设计)作为一种软件开发方法,它可以帮助我们设计高 ...

  9. DDD分层架构的三种模式

    引言 在讨论DDD分层架构的模式之前,我们先一起回顾一下DDD和分层架构的相关知识. DDD DDD(Domain Driven Design,领域驱动设计)作为一种软件开发方法,它可以帮助我们设计高 ...

随机推荐

  1. day6面向对象

    面向对象介绍(http://www.cnblogs.com/alex3714/articles/5188179.htm)     世界万物,皆可分类     世界万物,皆为对象     只要是对象,就 ...

  2. python开发之路Day17-算法设计(冒泡排序、选择排序、插入排序、二叉树)

    s12-20160514-day17 *:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: ...

  3. box-sizing定义的盒模型之间的区别

  4. elastucasearch基础理论以及安装

    一.elasticasearch核心概念 Near Realtime(NRT 近实时) Elasticsearch 是一个近实时的搜索平台.您索引一个文档开始直到它被查询时会有轻微的延迟时间(通常为1 ...

  5. ORACLE TO_CHAR(SYSDATE,'D')

    DDD是该天在一年内的第多少天,d是在一周内第几天,dd是一个月内的 DY :Day of week abbreviated Mon, Tue, Fri DAY :Day of week spelle ...

  6. spring boot配置使用fastjson

    一.前言 spring boot默认使用jackson来操作json数据,相比于jackson,fastjson更好用,功能也强大,所以这里记录一下在spring boot中配置使用fastjson的 ...

  7. MySQL集群原理详解

    1. 为什么需要分布式数据库2. MySQL Cluster原理3. MySQL Cluster的优缺点4. MySQL Cluster国内应用5. 参考资料 1. 为什么需要分布式数据库 随着计算机 ...

  8. 洛谷P2671 求和 [数论]

    题目传送门 求和 格式难调,题面就不放了. 分析: $ZYYS$的一道题. 很显然是大力推公式.我们分析一下题目,实际上限制条件就是:下标同奇偶且颜色相同的数,那么我们先拿这个公式$(x+z)*(nu ...

  9. BZOJ 1212 [HNOI2004]L语言 【AC自动机 + 背包】

    题目链接[http://www.lydsy.com/JudgeOnline/problem.php?id=1212] 题意:给你一些单词,然后给出一个没有标点的文本串S,都是小写字符.现在让你求用给出 ...

  10. [BZOJ5292][BJOI2018]治疗之雨(概率DP+高斯消元)

    https://blog.csdn.net/xyz32768/article/details/83217209 不难找到DP方程与辅助DP方程,发现DP方程具有后效性,于是高斯消元即可. 但朴素消元显 ...