ASP.NET Core依赖注入&AutoFac

1. 前言

关于IOC模式和DI技术,网上已经有很多相关的探讨,再次就不过多赘述了,只是简单介绍一下它们的概念

  • 控制反转(IoC/Inverse Of Control):   调用者将创建实例的控制权交给IOC容器,由容器创建,所以称为控制反转。
  • 依赖注入(DI/Dependence injection) :   容器创建好实例后再注入给调用者的过程称为依赖注入。依赖注入技术让我们的应用程序实现了松散耦合

.NetFramewok和.Net Core的区别之一就是.net core所有的实例都是通过依赖注入来创建的。下面介绍一下ASP.NET CORE中如何使用依赖注入

2.ASP.NET Core 中自带的DI方式

ASP.NET Core本身已经集成了一个轻量级的IOC容器,开发者只需要定义好接口后,在Startup.cs的ConfigureServices方法里使用对应生命周期的绑定方法即可

例:

            //注册数据库基础操作
            services.AddScoped(typeof(IBLLConstruct<>), typeof(BLLConstruct<>));
            //注册缓存操作
            services.AddTransient(typeof(ICacheContext), typeof(CacheContext));
            services.AddScoped(typeof(IAuth), typeof(LocalAuth));
            services.AddSingleton(typeof(IHttpContextAccessor), 
            typeof(HttpContextAccessor));
AddTransient:服务在每次请求时被创建
AddScoped:服务在每次请求时被创建,生命周期横贯整次请求
AddSingleton:顾名思义Singleton(单例),服务在第一次请求时被创建(或者当我们在ConfigureServices中指定创建某一实例并运行方法),其后的每次请求将沿用已创建服务

在这之后,我们便可以将服务通过构造函数注入或者是属性注入的方式注入到Controller,View(通过使用@inject),甚至是Filter中(以前的项目是使用Unity将依赖注入到Filter,个人感觉不如.net core中注入的简洁)。

3.构造函数获取实例

        //数据访问
        protected IBLLConstruct<UserInforMations> _repository { get; set; }
        //缓存
        protected ICacheContext _cache { get; set; }

        protected IAuth _auth { get; set; }

        public LoginService(IBLLConstruct<UserInforMations> repository, ICacheContext cache, IAuth auth)
        {
            this._cache = cache;
            this._repository = repository;
            this._auth = auth;
        }

流程大概就是这样,我们启动项目来看一下它的执行顺序

(1)容器创建实例

(2)构造函数获取实例

4.使用AutoFac实现扩展

除了ASP.NETCore自带的IOC容器外,我们还可以使用其他成熟的DI框架,如Autofac,StructureMap等(本人只用过Unity,Autofac)。

 (1)安装autofac

    

(2)创建容器并注册依赖

修改Startup.cs中ConfigureServices方法 不要忘了将ConfigureServices的返回值修改为IServiceProvider 

public IServiceProvider ConfigureServices(IServiceCollection services)
        {
            services.Configure<CookiePolicyOptions>(options =>
            {
                // This lambda determines whether user consent for non-essential cookies is needed for a given request.
                options.CheckConsentNeeded = context => true;
                options.MinimumSameSitePolicy = SameSiteMode.None;
            });


            services.AddMvc().SetCompatibilityVersion(CompatibilityVersion.Version_2_1);

            services.AddDbContext<DirectSellContext>(options =>
            options.UseSqlServer(Configuration.GetConnectionString("DBContext"), b => b.UseRowNumberForPaging()));

            //使用AutoFac进行注入
            return new AutofacServiceProvider(AutofacExt.InitAutofac(services));

        }
(3)在AutofacExt类中创建容器并注册依赖
        private static IContainer _container;
        public static IContainer InitAutofac(IServiceCollection services)
        {
            var builder = new ContainerBuilder();

            builder.RegisterType<GuidTransientAppService>().As<IGuidTransientAppService>();
            builder.RegisterType<GuidScopedAppService>().As<IGuidScopedAppService>().InstancePerLifetimeScope();
            builder.RegisterType<GuidSingletonAppService>().As<IGuidSingletonAppService>().SingleInstance();

            builder.Populate(services);

            _container = builder.Build();
            return _container;

        }

InstancePerLifetimeScope:同一个Lifetime生成的对象是同一个实例


SingleInstance:单例模式,每次调用,都会使用同一个实例化的对象;每次都用同一个对象;


InstancePerDependency:默认模式,每次调用,都会重新实例化对象;每次请求都创建一个新的对象;

(4)AutoFac批量注册服务

通过以上方式可以实现注入,但是我们每定义一个接口,都要在AutoFac中注册一次,可以使用RegisterAssemblyTypes来避免这种重复劳动

反射获取程序集

        /// <summary>
        /// 根据程序集名称获取程序集
        /// </summary>
        /// <param name="AssemblyName">程序集名称</param>
        /// <returns></returns>
        public static Assembly GetAssemblyByName(String AssemblyName)
        {
            return Assembly.Load(AssemblyName);
        }

批量注册

            //注册Service中的对象,Service中的类要以Service结尾,否则注册失败
            builder.RegisterAssemblyTypes(GetAssemblyByName("MyProject.Domain")).
                Where(a => a.Name.EndsWith("Service")).AsImplementedInterfaces();

这样就实现了Domain层中以Service结尾的接口批量注册 命名格式如下:

 

 

 

posted @ 2018-12-31 19:44  Quebra  阅读(2856)  评论(1编辑  收藏  举报