最近帮朋友做一个网站,基于.NET框架来做
虽然很多年不用.NET,不过上手起来还是快的。
临上线测试的时候,一直提示这个错误:System.Web.HttpException: 请求在此上下文中不可用
(1)原因分析
由于我的站点中为了方便了解用户信息,根据用ip从淘宝IP库获取用户IP,最近淘宝IP库不太稳定,经常提示502错误,为了提升用户体验,我将获取IP的地址信息采用异步的方式来获取并写入数据库。
为什么我没有用客户端去获取呢,因为我客户端通过异步请求获取客户端地址信息后,为了写入数据库,我还得调用本地的服务写入数据库。
为了省事,我就在我的主页面aspx的cs文件中获取IP信息并写入数据库,并把这个操作通过委托来实现异步。
原因就出在这里了!我的异步操作函数如下:
private void WriteClientInfos()
{
<strong>String strPrePage = Request.ServerVariables["HTTP_REFERER"] == null ? "" : Request.ServerVariables["HTTP_REFERER"].ToString();
String strCurPage = Request.Url.ToString();
String strIp = Request.ServerVariables["REMOTE_ADDR"].ToString();
String strUserAgent = Request.UserAgent.ToString();</strong>
ClientInfo clientInfo = AccessAnalyzeController.getClientInfo(
strIp,
strPrePage,
strCurPage,
strUserAgent);
LogUtil.WriteMsg("【Master WriteClientInfos】 after getClientInfo.");
AccessAnalyzeController.WriteClientInfo(clientInfo);
}
这是一段正常的写入数据库的操作。
关键问题出在最上面的四句,我把Request对象的数据在多线程中调用!
(2)解决方法
方式一:网上提供的方式是将IIS采用经典模式,而不是用集成模式,具体不解释了。
我最初采用这种方式,在本地测试没问题后上传到服务器上线运行,将服务器设置为经典模式,并重启后,通过查看log,我发现还是会报错误。
方式二:将Request对象的数据在Page_Init中先赋值,然后再启动线程,即可!
究其根本,是会话上下文不支持多线程,所以在多线程之前,先从回话中得到信息,然后再启动多线程操作即可。