什么是依赖注入?

我们以实际的例子来加以介绍
实体如下
  1. public class Product
  2. {
  3. public int ID { get; set; }
  4. public string Name { get; set; }
  5. public decimal Price { get; set; }
  6. }
EF的实现如下
  1. public class ProductContext: DbContext
  2. {
  3. public ProductContext(): base( "ProductContext")
  4. {
  5.  
  6. }
  7. public DbSet< Product> Products { get; set; }
  8. }
  9. public class ProductRepo_EF
  10. {
  11. private ProductContext _ctx = new ProductContext ();
  12. public IEnumerable<Models.Product > GetAll()
  13. {
  14. return _ctx.Products.ToList();
  15. }
  16.  
  17. public Models. Product GetProduct( int id)
  18. {
  19. return _ctx.Products.FirstOrDefault(x=>x.ID==id);
  20. }
  21. }
在controller中
  1. public class ProductsController : ApiController{
  2. //这一行是问题根源所在
  3. ProductRepository _repository = new ProductRepository();
  4.  
  5. public IEnumerable<Product> Get()
  6. {
  7. return _repository.GetAll();
  8. }
  9.  
  10. public IHttpActionResult Get(int id)
  11. {
  12. var product = _repository.GetByID(id);
  13. if (product == null)
  14. {
  15. return NotFound();
  16. }
  17. return Ok(product);
  18. }}
  1. 我们的productController是依赖于productRespository来提供数据的,也就是我们形象的说法,这个controller依赖于_respository的实现。那如果我们的repo发生变化,甚至我们将不准备采用EF针对sqlserver提供的方式,我们想更换针对mysql的实现,如果有很多controller都依赖于repo的实现,那代码的改动量将会很巨大,而且很容易犯错。这在真正的工业代码中将是无法想象的。 所以我们必须采用以来注入的松耦合实现方式。
  1. public interface IProductRepository{
  2. IEnumerable<Product> GetAll();
  3. Product GetById(int id);
  4. }
  5.  
  6. public class ProductRepository : IProductRepository{
  7.  
  8. //
  9. }
  10.  
  11. public class ProductsController : ApiController{
  12. private IProductRepository _repository;
  13.  
  14. public ProductsController(IProductRepository repository)
  15. {
  16. _repository = repository;
  17. }
  18.  
  19. }

利用构造函数向外界暴露依赖,这样在再创建不同的实例的时候,只需要提供不同的实现就可以了,在代码内部则不会发生改动。关于更多依赖注入的基础知识,可以自行搜索,实在太多。我们这里针对webapi的依赖注入。 在mvc以及webapi中,与业务逻辑打交道,那就肯定少不了controller对于业务类(比方说Repo)的依赖。那这里想要实现解耦,就要控制controller实例的创建,但是在mvc框架中controller的创建是由框架自行完成的..顿时觉得无从下手了。在mvc2.0中,如果使用DefaultControllerFactory的实现,创建controller实例是通过反射调用无参构造函数来实现的。那我们想要使用默认工厂来通过构造函数注入,显然是不现实的。只能整体更换整个工厂。而随着mvc框架的发展,到mvc4的时候,框架已经对于外部的以来注入实现已经相当友好了。默认工厂不再通过单一的无参构造函数反射创建实例,而是统一的通过IDependencyresolver接口提供的 IDependencyResolver.GetService(Type serviceType)
方法来暴露实现。在创建controller实例的时候,首先通过该方法去取得controller实例,如果为null在调用无参构造创建实例。 所以我们只需要使用外部IOC容器扩展针对IDependencyResolver的实现即可。
外部IOC容器常用的有 Unity Autofac Ninject 等等。我们这里采用AutoFac做为我们的实现。因为文档齐全,效率高,功能强大,也是主流IOC容器。
使用autofac可以自行创建,也可以采用autofac官方针对webapi的扩展。 通过Nuget安装 autofac 以及autofac.extentions.webapi。将自动有AutofacWebApiDependencyResolver实现。

  1. global.asax
  1. var builder = new ContainerBuilder();
  2. builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
  3. //仅仅这一个地方的改动,就达到了对数据库的无缝链接。
  4. builder.RegisterType<ProductRepo_EF>().As<IProductRepo>();
  5. var container = builder.Build();
  6. var resolver = new AutofacWebApiDependencyResolver(container);
  7. GlobalConfiguration.Configuration.DependencyResolver = resolver;
  1. 在这里我们使用了EFEF to Mysql两种repo实现,也就是通过两种数据库提供数据。
  2. 备注一下,使用EF To Mysql需要安装mysql数据库,以及在nuget包中

  1.  
  1. 并且在web.config中提供mysql以及sqlserver两种连接字符串即可。
  2. 这样 ,当我们需要更换不同的repo实现的时候只需要在
builder.RegisterType<ProductRepo_EF>().As<IProductRepo>();   
  1. 替换成
builder.RegisterType<ProductRepo_Mysql>().As<IProductRepo>(); 即可。
我们的代码改动仅仅只发生在注入的这一个地方。

AutoFac实现WebAPI依赖注入(EF以及Mysql)的更多相关文章

  1. net core WebApi——依赖注入Autofac

    目录 前言 Autofac 添加一个Util来随时调用 小结 代码地址 前言 周末加班,下午犯困,整理下之前鼓捣过的东西,看过我之前的webapi系列的读者知道,我之前试过Aspect,但是升级到3. ...

  2. AutoFac IoC DI 依赖注入

    AutoFac IoC DI 依赖注入 记录点点滴滴知识,为了更好的服务后来者! 一.为什么使用AutoFac? 之前介绍了Unity和Ninject两个IOC容器,但是发现园子里用AutoFac的貌 ...

  3. ASP.NET Core2使用Autofac实现IOC依赖注入竟然能如此的优雅简便

    初识ASP.NET Core的小伙伴一定会发现,其几乎所有的项目依赖都是通过依赖注入方式进行链式串通的.这是因为其使用了依赖注入 (DI) 的软件设计模式,代码的设计是遵循着“高内聚.低耦合”的原则, ...

  4. ASP.NET Core2使用Autofac实现IOC依赖注入竟然能如此的优雅简便(转载)

    原文地址:https://www.cnblogs.com/Andre/p/9604759.html 初识ASP.NET Core的小伙伴一定会发现,其几乎所有的项目依赖都是通过依赖注入方式进行链式串通 ...

  5. WebAPi使用Autofac实现依赖注入

    WebAPi依赖注入  使用记录 笔记 1.NuGet包安装 2.控制器加入构造函数 3.Global.asax  ----Application_Start 应用程序启动时 using Autofa ...

  6. Web API(六):使用Autofac实现依赖注入

    在这一篇文章将会讲解如何在Web API2中使用Autofac实现依赖注入. 一.创建实体类库 1.创建单独实体类 创建DI.Entity类库,用来存放所有的实体类,新建用户实体类,其结构如下: us ...

  7. ASP.NET Core搭建多层网站架构【9.1-使用Autofac代替原生的依赖注入】

    2020/01/30, ASP.NET Core 3.1, VS2019, Autofac.Extensions.DependencyInjection 5.0.1 摘要:基于ASP.NET Core ...

  8. 记录对依赖注入的小小理解和autofac的简单封装

    首先,我不是一个开发者,只是业余学习者.其次我的文化水平很低,写这个主要是记录一下当前对于这块的理解,因为对于一个低水平 的业余学习者来说,忘记是很平常的事,因为接触.应用的少,现在理解,可能过段时间 ...

  9. 依赖注入容器Autofac与MVC集成

    Autofac是应用于.Net平台的依赖注入(DI,Dependency Injection)容器,具有贴近.契合C#语言的特点.随着应用系统的日益庞大与复杂,使用Autofac容器来管理组件之间的关 ...

随机推荐

  1. Linux防火墙配置与管理(16)

    防火墙指的是一个由软件和硬件设备组合而成.在内部网和外部网之间.专用网与公共网之间的边界上构造的保护屏障.是一种获取安全性方法的形象说法,它是一种计算机硬件和软件的结合,使Internet与Intra ...

  2. Linux巩固记录(2) java项目的编译和执行

    由于要近期使用hadoop等进行相关任务执行,操作linux时候就多了 以前只在linux上配置J2EE项目执行环境,无非配置下jdk,部署tomcat,再通过docker或者jenkins自动部署上 ...

  3. centos7上编译安装mysql5.6

    注意,在做实验室统一关闭防火墙做的,在生产环境需要做防火墙规则的,大家要注意,做的时候尽量都是模仿生产环境的,比如服务一般都在/data/soft下面,尽量避免在/usr/local/下面. 安装编译 ...

  4. [Leetcode]134.加油站

    这一题是贪心不是模拟 是贪心不是模拟 是贪心不是模拟! 如果用模拟的做法会比较慢,也失去了做这一题的趣味了. 模拟的方法很简单,就是每一个加油站都做起点模拟一遍,试一下能不能完成一圈,能完成一圈就保存 ...

  5. static、final、static final的区别

    final: final可以修饰属性,方法,类,局部变量(方法中的变量) final修饰的属性的初始化可以在编译期,也可以在运行期,初始化后不能被改变. final修饰的属性跟具体对象有关,在运行期初 ...

  6. SQLServer中的cross apply和FOR XML PATH

    参考: FOR XML PATH:http://www.cnblogs.com/doubleliang/archive/2011/07/06/2098775.html cross apply:http ...

  7. sql server 2008 R2

    SQL SERVER 2008 R2序列号: 数据中心版:PTTFM-X467G-P7RH2-3Q6CG-4DMYB 开 发者 版:MC46H-JQR3C-2JRHY-XYRKY-QWPVM 企    ...

  8. Linux安装go语言开发包

    1.下载go语言安装包,eg:go1.7.1.linux-amd64.tar.gz2.安装go语言 $ cd /home/xm6f/dev $ tar -zxvf go1.7.1.linux-amd6 ...

  9. mysql 根据一张表更新另一张表

    between 是>= and <=,即包含两个边界

  10. JavaScript事件-this传递

    <!DOCTYPE html> <html> <head> <meta http-equiv="Content-Type" content ...