Repository仓储 UnitofWork

目录索引 

【无私分享:ASP.NET CORE 项目实战】目录索引

简介

  本章我们来创建仓储类Repository 并且引入 UnitOfWork

我对UnitOfWork的一些理解

   UnitOfWork 工作单元,对于这个的解释和实例,网上很多很多大神之作,我在这就不班门弄斧了,只是浅谈 一下个人理解,不一定对,希望大家指正:

  UnitOfWork 看了很多,个人理解就是 统一事务,在很多操作中我们可能是进行多项操作的,比如 同时保存两个表的数据,如果我们先保存一个表(SaveChanges()之后),再保存另一个表(SaveChanges())其实是两个不同的事务,通俗的讲,我们对数据库提交了两次数据。而UnitOfWork 就是 把本次操作(分别保存两张表)放在统一事务中,所有的CRUD完成后,统一提交数据,即保证数据的一致性,又减少了提交次数,提高性能。

  举个例子,我们之前的Repository的保存操作:

  Repository:

  public virtual bool Save(T entity)
  {
    _Context.Add(entity);
    return _Context.SaveChanges() > 0; 
  }

  Controller:

  public ActionResult Save()

  {

    var Users=new Users(){Id=1,UserName="张三"};

    var Logs=new Logs(){log="注册了用户张三"};

    _UserService.Save(Users);

    _LogService.Save(Logs);

  }

因为本身EF就是带事务的,但是_Context.SaveChanges()之后,事务就结束了,也提交了数据。所以,上面的例子 应该是开启了两次事务,提交了两次数据(一次是Users 一次是 Logs)

这显然是不合理的,首先不说提交两次数据性能的问题,如果用户注册失败,那么日志也不应该添加,应该回滚。但是,显然,上面没有这么做。

所以有了UnitOfWork。

使用UnitOfWork之后,代码应该是这样的:

  UnitOfWork:

  public bool Commit()
  {
    return _Context.SaveChanges() > 0; 
  }

  Repository:

  public virtual bool Save(T entity)
  {
    _Context.Add(entity);
  }

  Controller:

  public ActionResult Save()

  {

    var Users=new Users(){Id=1,UserName="张三"};

    var Logs=new Logs(){log="注册了用户张三"};

    _UserService.Save(Users);

    _LogService.Save(Logs);

    _UnitOfWork.Commit();

  }

  

UnitOfWork接口和实现类

我们在wkmvc.Core类库下,新建一个接口 IUnitOfWork:

 1 namespace wkmvc.Core
2 {
3 /// <summary>
4 /// Describe:工作单元接口
5 /// Author:yuangang
6 /// Date:2016/07/16
7 /// Blogs:http://www.cnblogs.com/yuangang
8 /// </summary>
9 public interface IUnitOfWork
10 {
11 bool Commit();
12 }
13 }

  IUnitOfWork 实现类 UnitOfWork:

 1 using System;
2 using wkmvc.Data;
3
4 namespace wkmvc.Core
5 {
6 /// <summary>
7 /// Describe:工作单元实现类
8 /// Author:yuangang
9 /// Date:2016/07/16
10 /// Blogs:http://www.cnblogs.com/yuangang
11 /// </summary>
12 public class UnitOfWork : IUnitOfWork, IDisposable
13 {
14 #region 数据上下文
15
16 /// <summary>
17 /// 数据上下文
18 /// </summary>
19 private ApplicationDbContext _Context;
20 public UnitOfWork(ApplicationDbContext Context)
21 {
22 _Context = Context;
23 }
24
25 #endregion
26
27 public bool Commit()
28 {
29 return _Context.SaveChanges() > 0;
30 }
31
32 public void Dispose()
33 {
34 if(_Context!=null)
35 {
36 _Context.Dispose();
37 }
38 GC.SuppressFinalize(this);
39 }
40 }
41 }

   这样,UnitOfWork 就完成了。下面我们来添加仓储:

仓储 IRepository、Repository

新建接口 IRepository 添加基本的操作方法:

 1 using System;
2 using System.Linq.Expressions;
3
4 namespace wkmvc.Core
5 {
6 /// <summary>
7 /// Describe:仓储接口
8 /// Author:yuangang
9 /// Date:2016/07/16
10 /// Blogs:http://www.cnblogs.com/yuangang
11 /// </summary>
12 /// <typeparam name="T">实体模型</typeparam>
13 public interface IRepository<T> where T : class
14 {
15 #region 单模型 CRUD 操作
16
17 /// <summary>
18 /// 增加一条记录
19 /// </summary>
20 /// <param name="entity">实体模型</param>
21 /// <param name="IsCommit">是否提交(默认提交)</param>
22 /// <returns></returns>
23 bool Save(T entity, bool IsCommit = true);
24
25 /// <summary>
26 /// 更新一条记录
27 /// </summary>
28 /// <param name="entity">实体模型</param>
29 /// <param name="IsCommit">是否提交(默认提交)</param>
30 /// <returns></returns>
31 bool Update(T entity, bool IsCommit = true);
32
33 /// <summary>
34 /// 增加或更新一条记录
35 /// </summary>
36 /// <param name="entity">实体模型</param>
37 /// <param name="IsSave">是否增加</param>
38 /// <param name="IsCommit">是否提交(默认提交)</param>
39 /// <returns></returns>
40 bool SaveOrUpdate(T entity, bool IsSave, bool IsCommit = true);
41
42 /// <summary>
43 /// 通过Lamda表达式获取实体
44 /// </summary>
45 /// <param name="predicate">Lamda表达式(p=>p.Id==Id)</param>
46 /// <returns></returns>
47 T Get(Expression<Func<T, bool>> predicate);
48
49 /// <summary>
50 /// 删除一条记录
51 /// </summary>
52 /// <param name="entity">实体模型</param>
53 /// <param name="IsCommit">是否提交(默认提交)</param>
54 /// <returns></returns>
55 bool Delete(T entity, bool IsCommit = true);
56
57 #endregion
58 }
59 }

   实现类 Repository :

  1 using Microsoft.EntityFrameworkCore;
2 using System;
3 using System.Linq;
4 using System.Linq.Expressions;
5 using wkmvc.Data;
6
7 namespace wkmvc.Core
8 {
9 /// <summary>
10 /// Describe:仓储实现类
11 /// Author:yuangang
12 /// Date:2016/07/16
13 /// Blogs:http://www.cnblogs.com/yuangang
14 /// </summary>
15 /// <typeparam name="T">实体模型</typeparam>
16 public abstract class Repository<T> : IRepository<T> where T : class
17 {
18 #region 数据上下文
19
20 /// <summary>
21 /// 数据上下文
22 /// </summary>
23 private ApplicationDbContext _Context;
24
25 /// <summary>
26 /// 工作单元
27 /// </summary>
28 UnitOfWork _UnitOfWork;
29
30 public Repository(ApplicationDbContext Context)
31 {
32 _Context = Context;
33 _UnitOfWork = new UnitOfWork(_Context);
34 }
35
36
37 #endregion
38
39 #region 单模型 CRUD 操作
40
41 /// <summary>
42 /// 增加一条记录
43 /// </summary>
44 /// <param name="entity">实体模型</param>
45 /// <param name="IsCommit">是否提交(默认提交)</param>
46 /// <returns></returns>
47 public virtual bool Save(T entity,bool IsCommit=true)
48 {
49 _Context.Add(entity);
50 if (IsCommit)
51 return _UnitOfWork.Commit();
52 else
53 return false;
54 }
55
56 /// <summary>
57 /// 更新一条记录
58 /// </summary>
59 /// <param name="entity">实体模型</param>
60 /// <param name="IsCommit">是否提交(默认提交)</param>
61 /// <returns></returns>
62 public virtual bool Update(T entity, bool IsCommit = true)
63 {
64 _Context.Attach(entity);
65 _Context.Entry(entity).State = EntityState.Modified;
66 if (IsCommit)
67 return _UnitOfWork.Commit();
68 else
69 return false;
70 }
71
72 /// <summary>
73 /// 增加或更新一条记录
74 /// </summary>
75 /// <param name="entity">实体模型</param>
76 /// <param name="IsSave">是否增加</param>
77 /// <param name="IsCommit">是否提交(默认提交)</param>
78 /// <returns></returns>
79 public virtual bool SaveOrUpdate(T entity, bool IsSave, bool IsCommit = true)
80 {
81 return IsSave ? Save(entity, IsCommit) : Update(entity, IsCommit);
82 }
83
84 /// <summary>
85 /// 通过Lamda表达式获取实体
86 /// </summary>
87 /// <param name="predicate">Lamda表达式(p=>p.Id==Id)</param>
88 /// <returns></returns>
89 public virtual T Get(Expression<Func<T, bool>> predicate)
90 {
91 return _Context.Set<T>().AsNoTracking().SingleOrDefault(predicate);
92 }
93
94
95 /// <summary>
96 /// 删除一条记录
97 /// </summary>
98 /// <param name="entity">实体模型</param>
99 /// <param name="IsCommit">是否提交(默认提交)</param>
100 /// <returns></returns>
101 public virtual bool Delete(T entity, bool IsCommit = true)
102 {
103 if (entity == null) return false;
104 _Context.Remove(entity);
105
106 if (IsCommit)
107 return _UnitOfWork.Commit();
108 else
109 return false;
110 }
111
112 #endregion
113
114
115 }
116 }

我们都添加了一个  bool IsCommit = true 参数,是为了方便,如果仅仅是进行一项操作,那就没必要使用 UnitOfWork 统一 直接 SaveChanages() 就行了

我们来看下使用:

  //第一种
  public bool TestOne()
  {
    var users = new SYS_USER() { UserName="张三",Account="zhangsan",Password="123456" };

    return _UserService.Save(users);
  }

  //第二种
  public bool TestTwo()
  {
    var users1 = new SYS_USER() { UserName = "张三", Account = "zhangsan", Password = "123456" };
    var users2 = new SYS_USER() { UserName = "李四", Account = "lisi", Password = "456789" };
    var users3 = new SYS_USER() { UserName = "王五", Account = "wangwu", Password = "321654" };

    _UserService.Save(users1);
    _UserService.Save(users2);
    _UserService.Save(users3);

    return _UnitOfWork.Commit();
  }

本篇主要是介绍用法,具体的解释清大家参考大神们的讲解,如有不对的地方希望指正。具体的实现代码,我们后面一起一点一点写。

希望跟大家一起学习Asp.net Core

刚开始接触,水平有限,很多东西都是自己的理解和翻阅网上大神的资料,如果有不对的地方和不理解的地方,希望大家指正!

虽然Asp.net Core 现在很火热,但是网上的很多资料都是前篇一律的复制,所以有很多问题我也暂时没有解决,希望大家能共同帮助一下!

原创文章 转载请尊重劳动成果 http://yuangang.cnblogs.com

Repository仓储 UnitofWork的更多相关文章

  1. 【无私分享:ASP.NET CORE 项目实战(第五章)】Repository仓储 UnitofWork

    目录索引 [无私分享:ASP.NET CORE 项目实战]目录索引 简介 本章我们来创建仓储类Repository 并且引入 UnitOfWork 我对UnitOfWork的一些理解  UnitOfW ...

  2. 关于 Repository和UnitOfWork 模式的关系

    本以为,关于这方面的理解,园子中的文章已经很多的了,再多做文章真的就“多做文章了”,但是最近发现,还是有必要的,首先,每个人对于同一事物的理解方式和出发点都是不同的,所以思考的方式得到结果也是不同的. ...

  3. 一步一步学EF系列 【7、结合IOC ,Repository,UnitOfWork来完成框架的搭建】

    前言 距离上一篇已经有段时间了,最近这段时间赶上新项目开发,一直没有时间来写.之前的几篇文章,主要把EF的基础都讲了一遍,这批文章就来个实战篇. 个人在学习过程中参考博客: Entity Framew ...

  4. MVC实用架构设计(三)——EF-Code First(1):Repository,UnitOfWork,DbContext

    前言 终于到EF了,实在不好意思,最近有点忙,本篇离上一篇发布已经一个多星期了,工作中的小迭代告一段落,终于有点时间来继续我们的架构设计了,在这里先对大家表示歉意. 其实这段时间我并不是把这个系列给忘 ...

  5. 从Entity Framework的实现方式来看DDD中的repository仓储模式运用

    一:最普通的数据库操作 static void Main(string[] args) { using (SchoolDBEntities db = new SchoolDBEntities()) { ...

  6. Repository 仓储,你的归宿究竟在哪?(三)-SELECT 某某某。。。

    写在前面 首先,本篇博文主要包含两个主题: 领域服务中使用仓储 SELECT 某某某(有点晕?请看下面.) 上一篇:Repository 仓储,你的归宿究竟在哪?(二)-这样的应用层代码,你能接受吗? ...

  7. Repository 仓储,你的归宿究竟在哪?(二)-这样的应用层代码,你能接受吗?

    写在前面 关于"Repository 仓储,你的归宿究竟在哪?"这个系列,本来是想写个上下篇,但是现在觉得,很有多东西需要明确,我也不知道接下来会写多少篇,所以上一篇的标题就改成了 ...

  8. DDD:Repository和UnitOfWork的生命周期问题

    UnitOfWork UnitOfWork是一种有状态的.用例级别的对象.如果不采用ORM是不会使用UnitOfWork模式的, Repository Repository是一种特殊的领域服务,因此是 ...

  9. Repository 仓储

    Repository 仓储 写在前面 首先,本篇博文主要包含两个主题: 领域服务中使用仓储 SELECT 某某某(有点晕?请看下面.) 上一篇:Repository 仓储,你的归宿究竟在哪?(二)-这 ...

随机推荐

  1. WordPress插件制作教程(一): 如何创建一个插件

    上一篇还是按照之前的教程流程,写了一篇WordPress插件制作教程概述,从这一篇开始就为大家具体讲解WordPress插件制作的内容.这一篇主要说一下插件的创建方法. 相信大家都知道插件的安装文件在 ...

  2. Python之路第十三天,高级(7)-详述数据库一对多,多对多表关系的设计以及如何查询

    一对多表设计和查询方法 #!/usr/bin/env python3 # Author: Zhangxunan from sqlalchemy import create_engine from sq ...

  3. Nginx 的 Location 配置指令块

    最近一段时间在学习 Nginx ,以前一直对 Nginx 的 Location 配置很头大,最近终于弄出点眉目.总结如下:nginx 配置文件,自下到上分为三种层次分明的结构: |    http b ...

  4. PHP中output control

    Output Control 函数可以让你自由控制脚本中数据的输出.它非常地有用,特别是对于:当你想在数据已经输出后,再输出文件头的情况.输出控制函数不对使用 header() 或 setcookie ...

  5. 段和RSEG用法

    RSEG是段选择指令,要想明白它的意思就要了解段的意思. 段是程序代码或数据对象的存储单位.程序代码放到代码段,数据对象放到数据段.段分两种,一是绝对段,一是再定位段.绝对段在汇编语言中指定,在用L5 ...

  6. QThread与其他线程间相互通信

    转载请注明链接与作者huihui1988 QThread的用法其实比较简单,只需要派生一个QThread的子类,实现其中的run虚函数就大功告成, 用的时候创建该类的实例,调用它的start方法即可. ...

  7. 常用财务软件:用友,金蝶,新中大,速达,管家婆,金算盘,远方,远光,金钥匙,润衡,浪潮,上海博科,易商,任我行,千方百剂,智管,小蜜蜂,SAP,ORACLE,SSA,QAD,MAPICS,JDE。

    常用财务软件:用友,金蝶,新中大,速达,管家婆,金算盘,远方,远光,金钥匙,润衡,浪潮,上海博科,易商,任我行,千方百剂,智管,小蜜蜂,SAP,ORACLE,SSA,QAD,MAPICS,JDE. 申 ...

  8. hdu 4502 吉哥系列故事——临时工计划_简单dp

    题意:        俗话说一分钱难倒英雄汉,高中几年下来,吉哥已经深深明白了这个道理,因此,新年开始存储一年的个人资金已经成了习惯,不过自从大学之后他不好意思再向大人要压岁钱了,只能把唯一的希望放到 ...

  9. Exploring TCP state machine by graphs

    States TCP includes 11 states, they are: LISTEN SYN_SENT SYN_RECV ESTABLISHED FIN_WAIT1 CLOSE_WAIT F ...

  10. ubuntu 下安装伪分布式 hadoop

    安装准备: (1)hadoop安装包:hadoop-1.2.1.tar.gz (2)jdk安装包:jdk-7u60-linux-i586.gz (3)要是须要eclipse开发的话 还须要eclips ...