似乎直到在控制器作用域中解析的元素上调用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 :(
}
}
}
这是故意的吗?如果是,是否有任何解决此问题的方法?
答案 0 :(得分:2)
您看到的行为是ASP.NET Core工作方式的一个不幸之处。我不清楚为什么微软会选择这种行为,但是它似乎是从具有确切行为的Web API的工作方式复制而来的。处置显然是在请求结束时完成的,但是由于某种原因,异步上下文已在此之前清除,因此无法在单个异步上下文中运行完整的请求。
您基本上有两个选择:
TestRepo
设为作用域,并将value
存储在私有字段中。某些DI容器实际上应用了第二种技术。例如,简单注入器使用基于环境状态的作用域,并在封面下使用AsyncLocal<T>
。当集成到ASP.NET Core中时,它将请求包装在适用于此范围的中间件中。这意味着从Simple Injector解析的所有Scoped组件都将在ASP.NET Core管道处置其服务之前被处置,并且这会在异步上下文仍然可用的情况下发生。