本系列目錄
Asp.Net DDD架構(gòu)淺談——整體框架說(shuō)明
Asp.Net DDD架構(gòu)淺談——領(lǐng)域劃分、倉(cāng)儲(chǔ)應(yīng)用、Services層定義
Asp.Net DDD架構(gòu)淺談——圖片上傳被芳、縮略裁剪
Asp.Net DDD架構(gòu)淺談——依賴注入Autofac
Asp.Net DDD架構(gòu)淺談——網(wǎng)站配置
控制反轉(zhuǎn)(Inversion of Control汤求,縮寫(xiě)為IoC)是開(kāi)發(fā)中的一種設(shè)計(jì)模式议慰,可以理解為面向接口編程速址,可以有效的降低耦合度。舉個(gè)例子逆粹,看我們的解決方案圖:
我們把Repsositories和Services的接口都定義在Steven.Domain里面募疮,而實(shí)現(xiàn)是單獨(dú)的項(xiàng)目Steven.Domain.Repositories和Steven.Domain.Services。
為什么要這么麻煩呢僻弹? 合在一起不好嗎阿浓?
合在一起的話?cǎi)詈隙忍撸绻獡QORM會(huì)很麻煩蹋绽,比如說(shuō)我現(xiàn)在在Steven.Domain.Repositories中用的是Dapper芭毙,而如果我想轉(zhuǎn)用EntityFramework,我只要新建一個(gè)項(xiàng)目Steven.Domain.EFRepositories卸耘,然后在新項(xiàng)目中實(shí)現(xiàn)接口即可退敦,對(duì)于Presentation層來(lái)說(shuō),沒(méi)有任何改變鹊奖,因?yàn)樵赑resentation層我們調(diào)用的是接口苛聘。
Autofac
Autofac是一個(gè)開(kāi)源的依賴注入框架涂炎,使用也非常簡(jiǎn)單忠聚,首先是通過(guò)NuGet安裝:
然后創(chuàng)建一個(gè)配置類DependencyConfig,在里面做好接口配置
public class DependencyConfig
{
public static void Register()
{
var builder = new ContainerBuilder();
builder.RegisterControllers(Assembly.GetCallingAssembly())//注冊(cè)mvc的Controller
.PropertiesAutowired();//屬性注入
builder.RegisterGeneric(typeof(Repository<>))
.As(typeof(IRepository<>))
.InstancePerRequest()//每次http請(qǐng)求
.PropertiesAutowired();//屬性注入
//Repositories的注入
builder.RegisterAssemblyTypes(typeof(UsersRepository).Assembly)
.Where(t => t.Name.EndsWith("Repository"))
.AsImplementedInterfaces()
.InstancePerRequest()
.PropertiesAutowired();
//Services的注入
builder.RegisterAssemblyTypes(typeof(FormsAuthenticationSvc).Assembly)
.Where(t => t.Name.EndsWith("Svc"))
.AsImplementedInterfaces()
.InstancePerRequest()
.PropertiesAutowired();
//Cache的注入唱捣,使用單例模式
builder.RegisterType<RedisCacheManager>()
.As<ICacheManager>()
.SingleInstance()
.PropertiesAutowired();
var container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
}
}
然后再Global.asax中配置两蟀,Application_Start網(wǎng)站啟動(dòng)的時(shí)候進(jìn)行配置:
DependencyConfig.Register();
如何使用呢?非常簡(jiǎn)單震缭, 我們打開(kāi)HomeController看下:
public class HomeController : WebSiteController
{
public IArticleRepository ArticleRepository { get; set; }
public ActionResult Index()
{
var list = ArticleRepository.GetAll();
return View(list);
}
public ActionResult Detail(string id)
{
var article = ArticleRepository.GetByIndex(id);
if (article == null)
{
return Redirect(Url.Home());
}
//update view count
ArticleRepository.UpdateViewCount(article.Id);
return View(article);
}
}
只需要定義一個(gè)屬性即可:public IArticleRepository ArticleRepository { get; set; } 是不是非常簡(jiǎn)單赂毯!
上面用到的是屬性注入的方式,回過(guò)頭了看下DependencyConfig類的定義,每個(gè)配置后面都有這么一句代碼:PropertiesAutowired()党涕,這就是表示使用屬性配置烦感。而在以前,我使用構(gòu)造函數(shù)注入膛堤,下面讓我們來(lái)看下構(gòu)造函數(shù)注入有什么不同:
首先是DependencyConfig的配置手趣,把PropertiesAutowired()刪掉即可,然后是使用到接口的地方肥荔,比如說(shuō)HomeController:
public class HomeController : WebSiteController
{
private readonly IArticleRepository ArticleRepository;
public HomeController(IArticleRepository articleRepository)
{
ArticleRepository = articleRepository;
}
public ActionResult Index()
{
var list = ArticleRepository.GetAll();
return View(list);
}
public ActionResult Detail(string id)
{
var article = ArticleRepository.GetByIndex(id);
if (article == null)
{
return Redirect(Url.Home());
}
//update view count
ArticleRepository.UpdateViewCount(article.Id);
return View(article);
}
}
屬性變成了私有變量绿渣,并且在HomeController的構(gòu)造函數(shù)中傳一個(gè)參數(shù)過(guò)來(lái)賦值。這種方式稍顯麻煩燕耿,個(gè)人意見(jiàn)中符。
注意事項(xiàng)
依賴注入的兩個(gè)類中不能互相依賴,比如說(shuō)我們?cè)贏Repository中用到了IBRepository誉帅,那么就不能再BRepository里面注入IARepository淀散,這樣會(huì)無(wú)限循環(huán),導(dǎo)致異常蚜锨。