7.1ASP.NET Core中的依赖注入
ASP.Net中Program.cs文件中,第一行var builder = WebApplication.CreateBuilder(args)
返回的类型为WebApplicationBuilder
类型,在该类型中Services属性为IServiceCollection
类型,也就是容器接口,一般把服务都注册到这个Service属性里面
varbuilder=WebApplication.CreateBuilder(args);
builder.Services.AddControllers();//将项目中的控制器及相关的服务注册到容器中
builder.Services.AddEndpointsApiExplorer();
builder.Services.AddSwaggerGen();//将Swagger相关服务注册到容器中
//当我们想注册服务的时候,主要把注册代码放到builder.Build()之前就可以
//注册代码的顺序不会影响程序运行的效果
varapp=builder.Build();
因为控制器被AddControllers()方法注册到了容器中,所以控制器中可以使用依赖注入的方式,在构造函数中依赖注入其他的类
publicclassMyService1
{
publicIEnumerable<string>GetNames()
{
returnnewstring[] { "Tom", "Zack", "Jack" };
}
}
//如果在var app = builder.Build();之前使用builder.Services.AddScoped<MyService1>();就可以像下面这样来注入服务了
publicclassTestController : ControllerBase
{
privatereadonlyMyService1myService1;
publicTestController(MyService1myService1)//构造函数,注入MyService1
{
this.myService1=myService1;
}
[HttpGet]
publicstringTest()
{
varnames=myService1.GetNames();
returnstring.Join(",", names);
}
}
低频率服务的另类注入方式
如果一个类中有多个操作方法,这些操作方法用到的服务都要使用构造函数来注入(所有的服务都得包含进来,不管用不用)。如果一个操作方法用到的服务特别耗费资源并且使用频率低,则可以在操作方法中通过参数注入的方式,实现在执行某个方法的时候才注入特定的服务。
publicstringTest([FromServices]MyService1myService1, stringname) //只有在调用Test的时候,才注入MyService1,且不需要在构造函数中有
{
varnames=myService1.GetNames();
returnstring.Join(",", names) +",hello:"+name;
}
//建议多数还是要才去构造函数的方式来注入
//[FromServices]只有控制器类的操作方法才能这样用
案例
一个解决方案中会包含多个项目,并且解决方案中会引用各个项目,每个项目都会用到很多被注入的服务,所以我们需要在解决方案中把设计到的服务都要注册进去,这样就会很麻烦。现在开发一个案例来简化这些操作。
//创建接口和类
publicinterfaceImoduleInitializer
{
publicvoidInitialize(IServiceCollectionservices);
}
publicstaticIServiceCollectionRunModuleInitializers(thisIServiceCollectionservices,IEnumerable<Assembly>assemblies)
{
//扫描所有程序集中实现了IModuleInitializer的类
foreach (varimplTypeinassemblies.SelectMany(asm=> asm.GetTypes()).Where(t=>!t.IsAbstract&&typeof(IModuleInitializer).IsAssignableFrom(t))
{
//创建了ImoduleInitializer类
varinitializer= (IModuleInitializer?)Activator.CreateInstance(implType);
initializer.Initialize(services);
}
returnservices;
}
- 首先创建一个类库项目,并创建接口IMyService
publicinterfaceIMyService
{
voidSayHello();
}
- 创建类库项目demo1,引用IMyService接口
publicclassCnService : IMyService
{
publicvoidSayHello()
{
Console.WriteLine("你好");
}
}
- 在demo1项目中创建IModuleInitializer的实现类ModuleInitializer
classModuleInitializer : IModuleInitializer
{
publicvoidInitialize(IServiceCollectionservices)
{
services.AddScoped<IMyService, CnService>();//将CnService注册为IMyService的实现服务
}
}
- 创建控制台程序,添加对上面项目的引用
ServiceCollectionservices=newServiceCollection();
varassemblies=ReflectionHelper.GetAllReferencedAssemblies();//获取用户的所有程序集,代码过长,没有在这里给出
services.RunModuleInitializers(assemblies);//扫描指定程序集中所有实现了IModuleInitializer的类,并调用Initialize方法
usingvarsp=services.BuildServiceProvider();
varitems=sp.GetServices<IMyService>();
foreach(variteminitems)
{
item.SayHello();
}
//并没有注册CnService服务但是可以使用