Voo*_*Voo 4 .net c# dependency-injection .net-core asp.net-core
似乎直到Dispose在控制器作用域中解析的元素上调用之前,执行上下文都不会保留。这可能是由于asp.net核心必须在本机代码和托管代码之间跳转,并在每次跳转时重置执行上下文。似乎在处置范围之前不再恢复正确的上下文。
以下内容演示了该问题-只需将其放入默认的asp.net核心示例项目中并注册TestRepo为临时依赖项即可。
调用时,GET api/values/我们在调用开始时在静态AsyncLocal中将当前任务的值设置为5。该值按预期方式流动,毫无问题。但是,在调用之后处置控制器及其依赖项时,AsyncLocal上下文已被重置。
[Route("api/[controller]")]
public class ValuesController : Controller
{
private readonly TestRepo _testRepo;
public ValuesController(TestRepo testRepo) => _testRepo = testRepo;
[HttpGet()]
public async Task<IActionResult> Get()
{
_testRepo.SetValue(5);
await Task.Delay(100);
var val = _testRepo.GetValue(); // val here has correctly 5.
return Ok();
}
}
public class TestRepo : IDisposable
{
private static readonly AsyncLocal<int?> _asyncLocal = new AsyncLocal<int?>();
public int? GetValue() => _asyncLocal.Value;
public void SetValue(int x) => _asyncLocal.Value = x;
public void Foo() => SetValue(5);
public void Dispose()
{
if (GetValue() == null)
{
throw new InvalidOperationException(); //GetValue() should be 5 here :(
}
}
}
Run Code Online (Sandbox Code Playgroud)
这是故意的吗?如果是,是否有任何解决此问题的方法?
您所看到的行为是ASP.NET Core工作方式的一个不幸之处。我不清楚为什么微软会选择这种行为,但是它似乎是从具有确切行为的Web API的工作方式复制而来的。处置显然是在请求结束时完成的,但是由于某种原因,异步上下文已在该点之前清除,因此无法在单个异步上下文中运行完整的请求。
您基本上有两个选择:
TestRepo作用域”并存储value在私有字段中。一些DI容器实际上应用了第二种技术。例如,Simple Injector在后台使用基于环境状态的作用域AsyncLocal<T>。当集成到ASP.NET Core中时,它将请求包装在适用于此范围的中间件中。这意味着从Simple Injector解析的所有Scoped组件都将在ASP.NET Core管道释放其服务之前被处置,并且这种情况在异步上下文仍然可用的情况下发生。
| 归档时间: |
|
| 查看次数: |
561 次 |
| 最近记录: |