我有这个类会触发一个方法而忘记它...唯一的问题是,如果调用的方法有一个HttpContext
它会抛出NullReferenceException
。
我的意思是我无法在Httpcontext
内使用ThreadPool.QueueUserWorkItem(dynamicInvokeShim, new TargetInfo(d, args));
,因为我得到NullReferenceException
是否有解决方法?
Httpcontext
的方法:
public static DataTable GetDataTable(string name)
{
return (DataTable)HttpContext.Current.Cache[name];
}
触发并忘记方法的方法:
using System;
using System.Threading;
namespace XGen.Kuapo.BLL
{
public class AsyncHelper
{
class TargetInfo
{
internal TargetInfo(Delegate d, object[] args)
{
Target = d;
Args = args;
}
internal readonly Delegate Target;
internal readonly object[] Args;
}
private static WaitCallback dynamicInvokeShim = new WaitCallback(DynamicInvokeShim);
public static void FireAndForget(Delegate d, params object[] args)
{
ThreadPool.QueueUserWorkItem(dynamicInvokeShim, new TargetInfo(d, args));
}
static void DynamicInvokeShim(object o)
{
try
{
TargetInfo ti = (TargetInfo)o;
ti.Target.DynamicInvoke(ti.Args);
}
catch (Exception ex)
{
// Only use Trace as is Thread safe
System.Diagnostics.Trace.WriteLine(ex.ToString());
}
}
}
}
答案 0 :(得分:6)
就像usr所说:HttpContext.Current
仅在执行请求的线程上可用,并且您启动的线程池线程不是同一个线程。
但是,如果要访问ASP.NET缓存,还可以使用HttpRuntime.Cache
,这是一个可从每个线程获得的静态属性。 (HttpContext.Current.Cache
只返回HttpRuntime.Cache
,因此您无需担心任何差异。)
请注意,建议将从HttpContext.Current
接收的HttpContext实例传递给另一个线程:在另一个线程运行时,与捕获的HttpContext对应的请求可能已经结束,因此您可能最终使用已损坏的HttpContext实例。
答案 1 :(得分:1)
是的,HttpContext.Current
仅在执行HTTP请求的线程上可用。一个简单的解决方法是将表达式(DataTable)HttpContext.Current.Cache[name]
的结果传递给线程,这样线程本身就不需要访问HttpContext
。或者,将HttpContext.Current
对象传递给线程。
如何将数据传递给线程或任务超出了这个问题的范围,但你会在网上找到很多关于它的信息。