c# – .net core DI中的异步提供程序
我只是想知道在DI期间是否可以进行异步/等待.
执行以下操作,DI无法解析我的服务. services.AddScoped(async provider => { var client = new MyClient(); await client.ConnectAsync(); return client; }); 以下作品完美无缺. services.AddScoped(provider => { var client = new MyClient(); client.ConnectAsync().Wait(); return client; }); 解决方法
解析依赖项时,Async / await没有意义,因为:
> Constructors can’t be asynchronous,和 这意味着所有涉及I / O的内容都应该推迟到构建对象图之后. 因此,MyClient不是注入连接的MyClient,而是在第一次使用时连接,而不是在创建时连接. UPDATE 由于您的MyClient不是应用程序组件而是第三方组件,这意味着您无法确保它“首次使用时连接[s]”. 然而,这应该不是问题,因为Dependency Inversion Principle已经告诉我们:
这意味着应用程序组件不应直接依赖于第三方组件,而应依赖于应用程序本身定义的抽象.作为Composition Root的一部分,可以编写实现这些抽象的适配器,并将应用程序代码调整到第三方库. 这样做的一个重要优点是,您可以控制应用程序组件使用的API,这是成功的关键,因为它允许连接问题完全隐藏在抽象背后. 以下是您的应用程序定制抽象可能如下所示的示例: public interface IMyAppService { Task<Data> GetData(); Task SendData(Data data); } 请注意,此抽象缺少ConnectAsync方法;这隐藏在抽象背后.例如,请查看以下适配器: public class MyClientAdapter : IMyAppService { private readonly Lazy<Task<MyClient>> connectedClient; public MyClientAdapter() { this.connectedClient = new Lazy<Task<MyClient>>(async () => { var client = new MyClient(); await client.ConnectAsync(); return client; }); } public async Task<Data> GetData() { var client = await this.connectedClient.Value; return await client.GetData(); } public async Task SendData(Data data) { var client = await this.connectedClient.Value; await client.SendData(data); } } 适配器从应用程序代码中隐藏有关连接的详细信息.它将MyClient的创建和连接包装在Lazy< T>中,它允许客户端只连接一次,而不管调用GetData和SendData方法的顺序,以及多少次. 现在,您可以让您的应用程序组件依赖于IMyAppService而不是MyClient,并以适当的生活方式将MyClientAdapter注册为IMyAppService. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |