本系列将使用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. Codeforces 946D Timetable(预处理+分组背包)

    题目链接:http://codeforces.com/problemset/problem/946/D 题目大意:有n个字符串,代表n天的课表,1表示这个时间要上课,0表示不要上课,一天在学校时间为第 ...

  2. Python学习笔记:个税起征点上调至5000,算一算少交多少税?

    一.旧税率表与新税率表比较 以前起征点是3500,2018年10月1日起起征点正式修改为5000,下面我们用Python来分别计算新旧个人所得税分别为多少? 二.旧的个人所得税 import sys ...

  3. Effective STL 学习笔记 39 ~ 41

    Effective STL 学习笔记 39 ~ 41 */--> div.org-src-container { font-size: 85%; font-family: monospace; ...

  4. HBase入门笔记--读性能优化

    一.前言 在生产环境使用HBase过程中,随着数据量的不断增加,查询HBase数据变得越来越慢,对于业务来说是不可用的,需要对读性能进行优化 二.问题定位 从hbase监控指标来看,发现FullGC次 ...

  5. 分析Windows的死亡蓝屏(BSOD)机制

    这篇文章本来是投Freebuf的,结果没过.就贴到博客里吧,图懒得发上来了 对于Windows系统来说,被人们视为洪水猛兽的蓝屏也是一种有利于系统稳定的机制.蓝屏其实是Windows系 统的一种自查机 ...

  6. **PHP删除数组中特定元素的两种方法array_splice()和unset()

    方法一: 复制代码代码如下: <?php$arr1 = array(1,3, 5,7,8);$key = array_search(3, $arr1); if ($key !== false)  ...

  7. SQL中的注释语句

    SQL中的注释分为单行注释和多行注释.顾名思义,单行注释就是对一行进行注释,多行注释就是同时对多行进行注释. 一.单行注释 SQL语句中的单行注释使用 -- create database datab ...

  8. Mybatis入门及Dao开发方式

    本节内容: Mybatis介绍 使用jdbc编程问题总结 Mybatis架构 Mybatis入门程序 Dao的开发方式 SqlMapConfig.xml文件说明 一.Mybatis介绍 MyBatis ...

  9. Dubbo的特点

    1.介绍 DUBBO有良好的连通性.健壮性.伸缩性.升级性.结合dubbo可以相对于单体系统提升系统整体的扩展性 2. 连通性 注册中心负责服务地址的注册与查找,相当于目录服务,服务提供者和消费者只在 ...

  10. python3下django连接mysql数据库

    1.安装pymysql pip install pymysql 有一点需要注意,有的系统(比如ubuntu16.04)同时安装了python2和python3,而比较新的django需要在python ...