我只是想知道在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,和
>对象图的构造应为simple,reliable and fast
这意味着所有涉及I / O的内容都应该推迟到构建对象图之后.
因此,MyClient不是注入连接的MyClient,而是在第一次使用时连接,而不是在创建时连接.
UPDATE
由于您的MyClient不是应用程序组件而是第三方组件,这意味着您无法确保它“首次使用时连接[s]”.
然而,这应该不是问题,因为Dependency Inversion Principle已经告诉我们:
the abstracts are owned by the upper/policy layers
这意味着应用程序组件不应直接依赖于第三方组件,而应依赖于应用程序本身定义的抽象.作为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.