一、前言

  在实际编写程序时,往往需要与数据库打交道,在单元测试中直接使用数据库又显得太重,如果可以方便的编写一些测试数据,这样更易于检测功能。如何模拟数据库行为便是本篇的主题。微软有教程说明Moq Entity Framework,需注意的是EF的版本必须是6以上。但在这篇教程中是直接使用DbContext,而自己的应用程序中都是用UnitOfWork模式。经过修改后也可以实现类似功能。

二、参考文献

https://msdn.microsoft.com/en-us/data/dn314429

三、采用UnitOfWork模式管理数据库

UnitOfWork

 public interface IDomainUnitOfWork : IDisposable
{
DbContext Db { get; }
//ImsDbContext dbContext { get; }
Task SaveChangesClientWinAsync();
Task SaveChangesDataBaseWinAsync(); void SaveChangesClientWin();
void SaveChangesDataBaseWin();
}

Repository接口

 public interface IDomainRepositoryAsync<T> where T : class
{ //Async Task<List<T>> GetAllAsync(Expression<Func<T, bool>> predicate = null); Task<T> SingleAsync(object primaryKey);
Task<T> SingleOrDefaultAsync(object primaryKey);
Task<bool> IsExistsAsync(Expression<Func<T, bool>> predicate = null); //同步
IQueryable<T> GetAll(Expression<Func<T, bool>> predicate = null);
T Single(object primaryKey);
T SingleOrDefault(object primaryKey);
bool IsExist(Expression<Func<T, bool>> predicate = null);
void Add(T entity);
void Update(T entity);
void Delete(T entity); }

Repository实现

public class DomainRepositoryAsync<T> : IDomainRepositoryAsync<T> where T : class
{
private readonly IDomainUnitOfWork _unitOfWork;
internal DbSet<T> dbSet; public DomainRepositoryAsync(IDomainUnitOfWork unitOfWork)
{
if (unitOfWork == null) throw new ArgumentNullException("unitOfWork");
this._unitOfWork = unitOfWork;
this.dbSet = _unitOfWork.Db.Set<T>();
}
public void Add(T entity)
{
dynamic obj = dbSet.Add(entity);
} public void Delete(T entity)
{
if (_unitOfWork.Db.Entry(entity).State == EntityState.Detached)
{
dbSet.Attach(entity);
}
dynamic obj = dbSet.Remove(entity);
} public async Task<List<T>> GetAllAsync(Expression<Func<T, bool>> predicate = null)
{ if (predicate != null)
{
return await this.dbSet.Where(predicate).ToListAsync();
} return await this.dbSet.ToListAsync();
} public async Task<bool> IsExistsAsync(Expression<Func<T, bool>> predicate = null)
{
var result = false;
if (predicate == null)
{
return result;
}
var query = await this.dbSet.Where(predicate).FirstOrDefaultAsync();
result = query == null ? false : true;
return result;
}
/// <summary>
/// 如果没有找到指定键元素,抛出异常.
/// </summary>
/// <param name="primaryKey">The primary key.</param>
/// <returns>Task&lt;T&gt;.</returns>
/// <exception cref="System.Collections.Generic.KeyNotFoundException"></exception>
public async Task<T> SingleAsync(object primaryKey)
{
T dbResult = null;
dbResult = await dbSet.FindAsync(primaryKey);
if (dbResult == null)
{
throw new KeyNotFoundException();
}
return dbResult;
} public async Task<T> SingleOrDefaultAsync(object primaryKey)
{
var dbResult = await dbSet.FindAsync(primaryKey);
return dbResult;
} public void Update(T entity)
{
this.dbSet.Attach(entity);
_unitOfWork.Db.Entry(entity).State = EntityState.Modified;
} public IQueryable<T> GetAll(Expression<Func<T, bool>> predicate = null)
{
if (predicate != null)
{
return this.dbSet.Where(predicate);
} return this.dbSet;
} public T Single(object primaryKey)
{
T dbResult = null;
dbResult = dbSet.Find(primaryKey);
if (dbResult == null)
{
throw new KeyNotFoundException();
}
return dbResult;
} public T SingleOrDefault(object primaryKey)
{
var dbResult = dbSet.Find(primaryKey);
return dbResult;
} public bool IsExist(Expression<Func<T, bool>> predicate = null)
{
var result = false;
if (predicate == null)
{
return result;
}
var query = this.dbSet.Where(predicate).FirstOrDefault();
result = query == null ? false : true;
return result;
}
}

DbContext

public class DurationDbContext : DbContext
{
public virtual DbSet<Department> Departments { get; set; }
public virtual DbSet<Duration> Durations { get; set; }
}

四、配置UnitTest

1、首先用Nuget安装moq

2、注意在DomainRepositoryAsync中有一个DbSet<T> dbSet,需要Moq的就是该类型,并且在DbContext中的必须加"virtual“关键字。

3、Moq代码

var unitOfWork = new Mock<IDomainUnitOfWork>();
var mockDepartment = SetupMockDbSet<Department>(DepartmentList);
var mockDuration = SetupMockDbSet<Duration>(DurationList);
unitOfWork.Setup(m => m.Db.Set<Department>()).Returns(mockDepartment.Object);
unitOfWork.Setup(m => m.Db.Set<Duration>()).Returns(mockDuration.Object);

在教程中说到了如何处理异步的查询操作,教程很详细,此处便不再重复,直接将代码Copy到单元测试工程中即可,再将重复的代码作为一个方法SetupMockDBSet。

 public static Mock<DbSet<T>> SetupMockDbSet<T>(List<T> dataList) where T : class
{
var data = dataList.AsQueryable();
var mockSet = new Mock<DbSet<T>>();
mockSet.As<IDbAsyncEnumerable<T>>()
.Setup(m => m.GetAsyncEnumerator())
.Returns(new TestDbAsyncEnumerator<T>(data.GetEnumerator())); mockSet.As<IQueryable<T>>()
.Setup(m => m.Provider)
.Returns(new TestDbAsyncQueryProvider<T>(data.Provider)); mockSet.As<IQueryable<T>>().Setup(m => m.Expression).Returns(data.Expression);
mockSet.As<IQueryable<T>>().Setup(m => m.ElementType).Returns(data.ElementType);
mockSet.As<IQueryable<T>>().Setup(m => m.GetEnumerator()).Returns(data.GetEnumerator());
return mockSet;
}

注意在配置UnitOfWork时,用Setup中是对IUnitOfWork的Db进行设置。接下来的实现方式与教程相同,不再重复。

利用Mocking Framework 单元测试Entity Framework的更多相关文章

  1. [Entity Framework] MySQL @ Entity Framework 6

    原文 [Entity Framework] MySQL @ Entity Framework 6 要让MySQL能够用EF6,我花了一点时间,在此记录一下 安装元件 在设定档加入Provider 安装 ...

  2. Entity Framework 教程——Entity Framework中的实体类型

    Entity Framework中的实体类型 : 在之前的章节中我们介绍过从已有的数据库中创建EDM,它包含数据库中每个表所对应的实体.在EF 5.0/6.0中,存在POCO 实体和动态代理实体两种. ...

  3. 【Entity Framework】 Entity Framework资料汇总

    Fluent API : http://social.msdn.microsoft.com/Search/zh-CN?query=Fluent%20API&Refinement=95& ...

  4. 基于Entity Framework 6的框架Nido Framework

    随着 Entity Framework 最新主版本 EF6 的推出,Microsoft 对象关系映射 (ORM) 工具达到了新的专业高度,与久负盛名的 .NET ORM 工具相比已不再是门外汉. EF ...

  5. 8天掌握EF的Code First开发之Entity Framework介绍

    返回<8天掌握EF的Code First开发>总目录 本篇目录 Entity Framework概要 什么是ORM Entity Framework简史 Entity Framework具 ...

  6. ADO.NET Entity Framework

    ADO.NET Entity Framework 是微软以 ADO.NET 为基础所发展出来的对象关系对应 (O/R Mapping) 解决方案, 早期被称为 ObjectSpace,现已经包含在 V ...

  7. 转载 8天掌握EF的Code First开发之Entity Framework介绍

    转载原地址:  http://www.cnblogs.com/farb/p/IntroductionToEF.html Entity Framework概要 Entity Framework是微软的O ...

  8. Entity Framework开发介绍

    一.Entity Framework概要 Entity Framework是微软的Object Relational Mapper(对象关系映射),也就是我们平常说的ORM,它可以让应用程序开发者将关 ...

  9. 深入了解Entity Framework框架及访问数据的几种方式

    一.前言 1.Entity Framework概要 Entity Framework是微软以ADO.NET为基础所发展出来的对象关系映射(O/R Mapping)解决方案.该框架曾经为.NET Fra ...

随机推荐

  1. win7安装ElasticSearch集群

    1.单节点安装请参考上篇博客 http://www.cnblogs.com/lianliang/p/7953754.html 2.集群的安装(这里模拟两个节点) 1)集群的安装,基于之前单节点的安装 ...

  2. (七)C语言之顺序结构

  3. [SQL Server创建视图时的注意点]

    创建视图的查询语句必须要遵守一定的限制 1. 要对某些列取别名,并保证列名的唯一 (具有相同的列名的表,在创建视图的时候,需要使用别名,表名.列名 也是不可以的) 当我们在通过新建视图来创建视图的话, ...

  4. 八十:memcached之安装与参数

    Memcached是一个高并发的内存键值对缓存系统,它的主要作用是将数据库查询结果,内容,以及其它一些耗时的计算结果缓存到系统内存中,从而加速Web应用程序的响应速度. 官网:http://memca ...

  5. Ubunut16.04 安装 Theano+GPU

    1. 更新NVIDIA显卡驱动   安装好系统之后首先在系统更新管理器中更新显卡驱动,如下图 点击Apply Changes 2. 安装numpy,scipy,theano pip安装即可 sudo ...

  6. JAVA学习记录<一>

    一: JAVA初体验: 1.JAVA简介: 2.环境搭建: 3:MyEclipse的使用简介: 4:程序的移植:项目的导入,导出. 5:学习JAVA的经验: 多写,多问,总结和复习!!!

  7. 开发单体web shop] 6. 商品分类和轮播广告展示

    目录 商品分类&轮播广告 商品分类|ProductCategory 需求分析 开发梳理 编码实现 轮播广告|SlideAD 需求分析 开发梳理 编码实现 福利讲解 源码下载 下节预告 商品分类 ...

  8. 从gopath到go mod的一次尝试

    windows下的尝试: gomod初尝试下载官方包1.11(及其以上版本将会自动支持gomod) 默认GO111MODULE=auto(auto是指如果在gopath下不启用mod)go mod h ...

  9. 串口使用和CSerial类

    1 串口通信的基本原理 串口通信中无论是写入串口还是读取串口,都是对缓冲区操作的.可以理解为写串口就是向输出缓冲区写入内容,读取串口就是从输入串口缓冲区读取内容.但是何时打开串口,何时发送数据,何时接 ...

  10. Android测试之查看package和activity名称的方法

    方法一:使用aapt    //aapt是sdk自带的一个工具,在sdk\builds-tools\目录下 1.命令行中切换到aapt.exe目录执行:aapt dump badging + 路径  ...