C#异步,多线程下的HttpContext丢失问题
1、思路概述
首先让我把大概的一个思路先说一遍吧。
我在一个页面中要同时调用两个接口,而我要给这些接口一些参数:就是我通过HttpContext.Current.Request.QueryString获取的URL地址上的参数。因为由于特殊需要,我需要把这写参数全部放到一个类中,然后让Page页面继承就可以直接调用其中的属性了。然后接受返回的参数,由于这两者的接口内部处理的比较复杂,做的处理比较多,再者我之前也是一直在学习,也看了博客园里写的一些知识,我想学以致用!:-)
是什么方法?当然是随处可见的异步了,异步的好处相信大家都知道!其中,经常挂在嘴边的就是增加吞吐量。至于异步是什么,博客园里面讲的好的太多了!我就不在这里赘述了。既然想到了那就立马执行
2、代码描述
首先是把URL上的parameter写到一个继承自Page类NewRulePage类:
public class NewRulePage : System.Web.UI.Page
public string Asid
{
get
{
string p = MyRequest.GetQueryString("adsid");
if (string.IsNullOrEmpty(p))
{
return "";
}
return p;
}
}
public string Aid
{
get
{
string a = MyRequest.GetQueryString("a");
if (string.IsNullOrEmpty(a))
{
return "";
}
return a;
}
}
...
}
接着是触发调用接口事件(其他无关紧要的代码忽略不看)
protected void Btn_submit_Click(object sender, EventArgs e)
...
Action<int, string, string, string, string> pa_func = SelectPaService;
//开始异步
var context = HttpContext.Current;
IAsyncResult async = pa_func.BeginInvoke(newid, IP, xfw, rep, cityid, null, null);
pa_func(newid, IP, xfw, rep, cityid);
SelectSsService(newid, IP, xfw, rep);
pa_func.EndInvoke(async);
//异步结束
...
//调用接口1
private void SelectPaService(int newid, string IP, string xfw, string rep, string cityid)
//一系列的调用NewRulePage的属性的操作
...
//调用接口2
private void SelectSsService(int newid, string IP, string xfw, string rep)
//一系列的调用NewRulePage的属性的操作
...
3、出现问题
结果运行发现报错:未将对象引用设置到对象的实例
有趣的事还在后面,当我继续往下断点调试的时候,在异步期间的SelectSsService方法却能正常执行并得到返回结果,而SelectSsService这个方法也是获取URL参数。那为什么不同样报错呢?为什么HttpContext这个对象会在SelectPaService方法中不存在?这个Http上下文不应该”处处可见“么?百般无奈,我只好带着这些疑问在万能的度娘搜索一下看有没有解释的,无奈,讲HttpContext不少,但是这方面的却不多!囧
无奈,自己处理吧!!原理我先丢着暂时不管,咱遇到什么问题就解决什么问题;Go Go Go
4、解决问题
现在的问题是什么?
自然是在SelectPaService方法中HttpContext丢失了,那么我想办法恢复HttpContext不就行了吗?或者伪造一个?(说法不好 哈哈~) IAsyncResult异步我们知道,实质上其实就是izai运行期间开启了两个线程工作,所以我们可以将上面异步改成线程模式调用来说明问题:
Thread thread = new Thread(new ThreadStart(delegate(){
SelectPaService(newid, IP, xfw, rep, cityid);
}));
ta.Start();
SelectSsService(newid, IP, xfw, rep);
毫无疑问,出现一样的错误,主线程中能访问HttpContext,而子线程的丢失了,虽然如此,之前我说了,可以恢复一个或伪造一个!最直接的办法怎么做?我想你们应该都知道了,就是直接在异步(子线程开启之前)给调用的方法传一个HttpContext对象不就有了么? 为了印证想法,我还是运行一下:
Action<int, string, string, string, string, HttpContext> pa_func = SelectPaService;
////开始异步
var context = HttpContext.Current;
IAsyncResult async = pa_func.BeginInvoke(newid, IP, xfw, rep, cityid, context, null, null);
pa_func(newid, IP, xfw, rep, cityid, context);
SelectSsService(newid, IP, xfw, rep);
pa_func.EndInvoke(async);
运行结果一切OK!!!!值得高兴~~~∩_∩
5、原理补充说明
虽说问题解决了,但是不知道原理,下次出了一个类似的问题估计还是够吃力!所以想在园子里找找有没有讲这方面的。结果还真找到我需要的了(搜索关键字很重要啊~~~o(︶︿︶)o ),再此来分享一下:fish-li大神的文章:HttpContext.Current并非无处不在 这里就详细讲了为什么会丢失,以及怎么做(里面就讲到了我现在用的通过传参的方式)~~顺便再次膜拜一下园子里的大神们!!! Over~