1. 程式人生 > >使用.net core 自帶DI框架實現 延遲加載

使用.net core 自帶DI框架實現 延遲加載

exception block example test cep request 實現 輸出 resolve

在某些情況,我們希望能延遲一個依賴的初始化。如果使用的是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中獲取到對應得依賴實例。

使用.net core 自帶DI框架實現 延遲加載