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