Using StructureMap DI and Generic Repository
In this post, i will show how to use generic repository and dependency injection using structuremap. I will be using LINQ to SQL.
Generic Repository
The interface for the generic repository is like this:
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
|
public interface IRepository<T> where T : class { void Create(T entity); void Update(T entity); void Delete(T entity); void Copy(T source, T target); void Flush(); T Get(Expression<Func<T, bool>> predicate); IEnumerable<T> GetAll(); System.Data.Linq.Table<T> Table { get; } int Count(Expression<Func<T, bool>> predicate); IEnumerable<T> Fetch(Expression<Func<T, bool>> predicate); IEnumerable<T> Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order); IEnumerable<T> Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order, int skip, int count); } |
Now lets go to the implementation of the generic repository interface.
|
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
|
public class Repository<T> : IRepository<T> where T : class { protected IDataContextFactory _dataContextFactory; public Repository(IDataContextFactory dataContextFactory) { _dataContextFactory = dataContextFactory; } public System.Data.Linq.Table<T> Table { get { return _dataContextFactory.Context.GetTable<T>(); } } IEnumerable<T> IRepository<T>.GetAll() { return Table.ToReadOnlyCollection(); } #region IRepository<T> Members void IRepository<T>.Create(T entity) { Table.InsertOnSubmit(entity); _dataContextFactory.SaveAll(); } void IRepository<T>.Update(T entity) { if (Table.GetOriginalEntityState(entity) == null) { Table.Attach(entity); } _dataContextFactory.Context.Refresh(System.Data.Linq.RefreshMode.KeepCurrentValues, entity); _dataContextFactory.SaveAll(); } void IRepository<T>.Delete(T entity) { Table.DeleteOnSubmit(entity); _dataContextFactory.SaveAll(); } T IRepository<T>.Get(Expression<Func<T, bool>> predicate) { return Table.FirstOrDefault(predicate); } int IRepository<T>.Count(Expression<Func<T, bool>> predicate) { return Table.Count(predicate); } public virtual IQueryable<T> Fetch(Expression<Func<T, bool>> predicate) { return Table.Where(predicate); } public virtual IQueryable<T> Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order) { var orderable = new Orderable<T>(Fetch(predicate)); order(orderable); return orderable.Queryable; } public virtual IQueryable<T> Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order, int skip, int count) { return Fetch(predicate, order).Skip(skip).Take(count); } IEnumerable<T> IRepository<T>.Fetch(Expression<Func<T, bool>> predicate) { return Fetch(predicate).ToReadOnlyCollection(); } IEnumerable<T> IRepository<T>.Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order) { return Fetch(predicate, order).ToReadOnlyCollection(); } IEnumerable<T> IRepository<T>.Fetch(Expression<Func<T, bool>> predicate, Action<Orderable<T>> order, int skip, int count) { return Fetch(predicate, order, skip, count).ToReadOnlyCollection(); } #endregion } |
If you notice ToReadOnlyCollection() this is an extension method of the IEnumerable. This is defined as:
|
1
2
3
4
5
6
|
using System.Collections.ObjectModel; public static IList<T> ToReadOnlyCollection<T>(this IEnumerable<T> enumerable) { return new ReadOnlyCollection<T>(enumerable.ToList()); } |
I think thats if for our generic repository. To use it, for example if we have a Blog model on our LINQ to SQL we can use the repository like:
|
1
|
IRepository<Blog> BlogRepository = new Repository<Blog>(); |
Using StructureMap Dependency Injector
Before reading this, I assume that the reader has knowledge already about dependency injection.
First we have to reference the StructureMap DLL. It can be downloaded on the structuremap website.
Then we have to create the Registry Class. This class will configure the dependencies. This class should be of type StructureMap.Configuration.DSL.Registry
The Registry class may look like this:
|
1
2
3
4
5
6
7
8
|
public class ApplicationRegistry : Registry { public ApplicationRegistry() { For(typeof(IRepository<Blog>)).Use(typeof(Repository<Blog>)); For(typeof(IDataContextFactory)).Use(typeof(DataContext)); } } |
After configuring the dependency, we will going to add our Registry to the StructureMap configuration. Usually, this will be on a Bootstrapper class that looks like this:
|
1
2
3
4
5
6
7
8
9
10
|
public static class Bootstrapper { public static void ConfigureStructureMap() { ObjectFactory.Initialize(x => { x.AddRegistry(new ApplicationRegistry()); }); } } |
This code tells the application to use the dependency injection as configure on the registry class. This should be called on program startup, the main method on console application or on global.asax on web application.
|
1
|
Bootstrapper.ConfigureStructureMap(); |
To get the instance of the repository:
|
1
|
IRepository<Blog> Repository = ObjectFactory.GetInstance<IRepository<Blog>>(); |
That should be it. Its ready.
If you have questions, please leave comments. I would be happy to answer.
If you want to create test repository that give predictable data, you should have another configuration for your test. Another boot strapper class perhaps but this is for test specific repository configuration. And on your test project call the test boot strapper not the original one.
Or just use mocking for testing like my code below using MOQ:
[TestMethod] public void GetUserByIDTest() { User user = new User() { UserID = 1, UserName = “mcxiand”, Password = “password”, }; var userid = 1; //initialize mock var mock = new Mock<IRepository>(); //see if the method in something you have mocked has been called by using Verify(). mock.Setup(x => x.Get(It.IsAny<Expression<Func>>())).Returns(user);
//Initialize Service service = new UserService(mock.Object); //call the method to test var u = service.GetUser(userid);
//Assert Assert.AreEqual(u.UserID, userid, “not equal”); }
http://mcxiand.wordpress.com/2010/05/13/using-structuremap-di-and-generic-repository/
Using StructureMap DI and Generic Repository的更多相关文章
- Generic repository pattern and Unit of work with Entity framework
原文 Generic repository pattern and Unit of work with Entity framework Repository pattern is an abstra ...
- EF6 CodeFirst+Repository+Ninject+MVC4+EasyUI实践(完)
前言 这一篇是本系列的最后一篇,虽然示例讲到这里就停止呢,但对于这些技术的学习远不能停止.虽然本示例讲的比较基础,但是正如我第一篇说到的,这个系列的目的不是说一些高端的架构设计,而是作为一个入门级,对 ...
- Follow me to learn what is repository pattern
Introduction Creating a generic repository pattern in an mvc application with entity framework is th ...
- Using the Repository Pattern with ASP.NET MVC and Entity Framework
原文:http://www.codeguru.com/csharp/.net/net_asp/mvc/using-the-repository-pattern-with-asp.net-mvc-and ...
- [转]Using the Repository Pattern with ASP.NET MVC and Entity Framework
本文转自:http://www.codeguru.com/csharp/.net/net_asp/mvc/using-the-repository-pattern-with-asp.net-mvc-a ...
- Using the Repository and Unit Of Work Pattern in .net core
A typical software application will invariably need to access some kind of data store in order to ca ...
- 单元操作和仓储模式 repository+unitOfWork
仓储和工作单元模式是用来在数据访问层和业务逻辑层之间创建一个抽象层.应用这些模式,可以帮助用来隔离你的程序在数据存储变化. 在数据源层和业务层之间增加一个repository层进行协调,有如下作用:1 ...
- 6.在MVC中使用泛型仓储模式和依赖注入实现增删查改
原文链接:http://www.c-sharpcorner.com/UploadFile/3d39b4/crud-operations-using-the-generic-repository-pat ...
- MVC中使用泛型仓储模式和依赖注入
在ASP.NET MVC中使用泛型仓储模式和依赖注入,实现增删查改 原文链接:http://www.codeproject.com/Articles/838097/CRUD-Operations-Us ...
随机推荐
- varnish缓存安装使用
varnish PDF http://files.cnblogs.com/jimingsong/varnish.pdf 目前介绍CentOS6.4-64位系统 yum安装varnish: 配置varn ...
- EF中用Newtonsoft.Json引发的循环引用问题
描述: 1.双向关系表a->b b->aList 2.在查询a引用b以后 3.用Newtonsoft.Json 去tojsonstring 4.一个只有6条数据的json串 出现了一屏幕字 ...
- Redis和Memcached的区别【转】
如果简单地比较Redis与Memcached的区别,大多数都会得到以下观点:1 Redis不仅仅支持简单的k/v类型的数据,同时还提供list,set,hash等数据结构的存储.2 Redis支持数据 ...
- 拓扑序+dp Codeforces Round #374 (Div. 2) C
http://codeforces.com/contest/721/problem/C 题目大意:给你有向路,每条路都有一个权值t,你从1走到n,最多花费不能超过T,问在T时间内最多能访问多少城市? ...
- android把图片 视频 保存到相册
//android把图片文件添加到相册 ContentResolver localContentResolver = getContentResolver(); ContentValues local ...
- sql server中单引号拼接字符串(书写错误会出现错误"浮点值 XXXX 超出了计算机表示范围(8 个字节)。“XX”附近有语法错误。")
" ' "(单引号)的运用:在sql server中,两个" ' "(单引号)在拼接字符串的情况下运用,就是表示拼接上了一个" ' "单引号 ...
- win8上cmder文字重叠问题
1.用过ubuntu上的bash,zsh后发现win自带的cmd弱爆了,在网上搜索后找到了代替品cmder,下载安装后好发现中文错位的问题, 状况如下: 修复方法:把设置里面的Monospace选项勾 ...
- LightOJ 1030 Discovering Gold 数学期望计算
题目大意:给出长度为n的一条隧道,每个位置都有一定数量的财宝.给你一枚骰子,roll到几点就前进几步,如果即将到达的地方超过了这条隧道长度,就重新roll一次,走到n点结束.求这个过程能收获多少财宝. ...
- 推荐 iOS 网站:
0. https://developer.apple.com/videos/1. http://www.raywenderlich.com2. http://nshipster.com3. http: ...
- 用python计算md5,sha1,crc32
Linux下计算md5sum,sha1sum,crc: 命令 输出 $md5sum hello f19dd746bc6ab0f0155808c388be8ff0 hello $sha1sum hel ...