首先,我正在尝试为产品构建一个插件 - 所以环境不是我的,我对正在发生的事情没有太多的可见性。据我所知,系统运行的是.NET Framework 4.6.2,我试图确保我所有的依赖项都与系统上的内容相匹配。
我看到的问题是,当我发出await PostAsync(...)
请求时,我得到一个NullReferenceException
。这是100%可重复的。但是,如果我使用PostAsync(...).Result
发出完全相同的请求,一切正常。
鉴于我可以在解析响应时使用await
关键字,并且它仍然可以正常工作,这变得更加奇怪。
我想使用await
关键字以"正确"的方式执行此操作,但它不起作用。从我读过的所有内容来看,这可能是一个依赖问题,但我不知道在哪里可以找到它。我已经花了 3 天的时间,当我终于发现使用.Result
"修复"我的问题时。
相关代码:
private async Task<T> aFunction<T>(string path, Dictionary<string, string> data) {
var content = new FormUrlEncodedContent(data);
var response = client.PostAsync(path, content).Result; // This works
// var response = await client.PostAsync(path, content); // This does not
response.EnsureSuccessStatusCode();
var result = await response.Content.ReadAsStringAsync();
var parsed = JsonConvert.DeserializeObject<T>(result);
return parsed;
}
NRE 的堆栈跟踪:
Stack trace:
at System.Web.ThreadContext.AssociateWithCurrentThread(Boolean setImpersonationContext)
at System.Web.HttpApplication.OnThreadEnterPrivate(Boolean setImpersonationContext)
at System.Web.LegacyAspNetSynchronizationContext.CallCallbackPossiblyUnderLock(SendOrPostCallback callback, Object state)
at System.Web.LegacyAspNetSynchronizationContext.CallCallback(SendOrPostCallback callback, Object state)
at System.Threading.Tasks.AwaitTaskContinuation.RunCallback(ContextCallback callback, Object state, Task& currentTask)
--- End of stack trace from previous location where exception was thrown ---
at System.Threading.Tasks.AwaitTaskContinuation.<>c.<ThrowAsyncIfNecessary>b__18_0(Object s)
at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
at System.Threading.QueueUserWorkItemCallback.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem()
at System.Threading.ThreadPoolWorkQueue.Dispatch()
我会尝试使用这里列出的一些建议。
特别是尝试在web.config
中添加以下appSetting
:
<add key="aspnet:UseTaskFriendlySynchronizationContext" value="true" />
以及以下httpRuntime
system.web
元素:
<system.Web>
<httpRuntime targetFramework="4.5" />
</system.Web>
由于托管插件的 ASPX 页面不是异步的,因此我无法在处理程序中使用异步方法。我最终必须确保使用同步调用而不是异步调用。
为了证明这一点,添加线条后(马丁推荐(
<add key="aspnet:UseTaskFriendlySynchronizationContext" value="true" />
并使用 await 调用,我的插件在发出请求后开始挂起。
这个SO页面:异步死锁?,向我暗示我的插件所在的页面没有正确设置异步功能。我尝试添加RegisterAsyncTask,并在日志中看到一条新消息:
InvalidOperationException:此操作要求页面是异步的(Async 属性必须设置为 true(。