使用.net core 自带DI框架实现 延迟加载
在某些情况,我们希望能延迟一个依赖的初始化。如果使用的是autofac,我们可以通过注入Lazy
我们对 autofac GitHub上提供的一个例子进行进行简单改造,跑起来看看。
原Example的链接https://github.com/autofac/Examples/tree/master/src/AspNetCoreExample
微改后的代码
[Route("api/[controller]")]
public class ValuesController : Controller
{
private readonly Lazy<IValuesService> _valuesService;
public ValuesController(Lazy<IValuesService> valuesService)
{
_valuesService = valuesService;
}
// GET api/values
[HttpGet]
public IEnumerable<string> Get()
{
// Kestrel模式下这里会输出false,实例尚未创建
Console.WriteLine(_valuesService.IsValueCreated);
// 调用Lazy<T>的Value属性才真正创建实例
return this._valuesService.Value.FindAll();
}
}
直到目前core2.1版本,自带的DI依旧未支持延迟加载,如果我们尝试在使用自带DI的情况下套用上述代码,会得到一个异常,例如:
An unhandled exception occurred while processing the request.
InvalidOperationException: Unable to resolve service for type 'System.Lazy`1[WebApplication9.Services.IValuesService]' while attempting to activate 'WebApplication9.Controllers.ValuesController'.
Microsoft.Extensions.DependencyInjection.ActivatorUtilities.GetService(IServiceProvider sp, Type type, Type requiredBy, bool isDefaultParameterRequired)
如何利用core自带的DI实现呢?如果我们尝试百度,可能会搜到类似下面的答案。
services.AddTransient(typeof(Lazy<>));
那么这样的做法是否能解决我们的问题呢,为了简化演示代码。我们创建一个控制台程序并引用Microsoft.Extensions.DependencyInjection。
class Program
{
static void Main(string[] args)
{
var services = new ServiceCollection();
services.AddScoped<ITestService, TestService>();
services.AddTransient(typeof(Lazy<>));
var serviceProvider = services.BuildServiceProvider();
using (var scope = serviceProvider.CreateScope() )
{
var service = scope.ServiceProvider.GetService<Lazy<ITestService>>();
// 这边令人遗憾地输出了true,也就是说,这种方式的延迟注入是失败的
Console.WriteLine(service.IsValueCreated);
}
}
}
在查阅Stack Overflow的时候,我看到了这样的解决方案,感觉还是挺简单实用的,分享给大家。
原贴地址:https://stackoverflow.com/questions/44934511/does-net-core-dependency-injection-support-lazyt
public class LazyLoader<T> : Lazy<T>
{
public LazyLoader(IServiceProvider sp) : base(sp.GetRequiredService<T>)
{
}
}
class Program
{
static void Main(string[] args)
{
var services = new ServiceCollection();
services.AddScoped<ITestService, TestService>();
// services.AddScoped(typeof(Lazy<>), typeof(LazyLoader<>)); 也可以,区别不大
services.AddTransient(typeof(Lazy<>), typeof(LazyLoader<>));
var serviceProvider = services.BuildServiceProvider();
using (var scope = serviceProvider.CreateScope())
{
var service = scope.ServiceProvider.GetService<Lazy<ITestService>>();
Console.WriteLine(service.IsValueCreated); // 输出false
// 下面输出true,延迟注入的对象和正常注入的对象,本质上不会有差别
Console.WriteLine(service.Value == scope.ServiceProvider.GetService<ITestService>());
}
}
}
实现原理比较简单,在LazyLoader中注入ServiceProvider,调用父类的Value属性时会执行委托,从ServiceProvider中获取到对应得依赖实例。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?