博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
为什么获取的System.Web.HttpContext.Current值为null,HttpContext对象为null时如何获取程序(站点)的根目录...
阅读量:6385 次
发布时间:2019-06-23

本文共 4161 字,大约阅读时间需要 13 分钟。

ASP.NET提供了静态属性System.Web.HttpContext.Current,因此获取HttpContext对象就非常方便了。也正是因为这个原因,所以我们经常能见到直接访问System.Web.HttpContext.Current的代码:

1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Web; 5  6 namespace Test_HttpContext.Current 7 { 8     public class Test 9     {10 11         public Test()12         {13             string file =System.Web.HttpContext.Current.Request.MapPath("~/App_Data/1.xml");14 15             string text = System.IO.File.ReadAllText(file);16 17             //..........其它的操作18         }19 20         // 或者在一些方法中直接使用HttpContext.Current21         public void Test_1()22         {23             string url = System.Web.HttpContext.Current.Request.RawUrl;24 25             string username = System.Web.HttpContext.Current.Session["username"].ToString();26 27             string value = (string)System.Web.HttpContext.Current.Items["key"];28         }29 30         // 甚至还设计成静态属性31         public static string Test_232         {33             get34             {35                 return (string)System.Web.HttpContext.Current.Items["XXX"];36             }37         }38 39         /// 40         /// 获取文件绝对路径41         /// 42         /// 文件名称43         /// 
44 public string Test_3(string fileName)45 {46 return System.Web.HttpContext.Current.Server.MapPath("~/Log" + fileName);47 }48 49 }50 }

上面的这些代码这样写真的没有问题吗?

答案是否定的

请看下面的验证:

我们先来看看HttpContext到底存储在哪里:

1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Web; 5 using System.Web.UI; 6 using System.Web.UI.WebControls; 7  8 namespace Test_HttpContext.Current 9 {10     public partial class WebForm1 : System.Web.UI.Page11     {12         protected void Page_Load(object sender, EventArgs e)13         {14 15             HttpContext context1 = System.Web.HttpContext.Current;16 17             HttpContext context2 = System.Runtime.Remoting.Messaging.CallContext.HostContext as HttpContext;  //当前(请求)线程上下文18 19             bool isEqual = object.ReferenceEquals(context1, context2);20 21             Response.Write("context1与context2是否相同的实例:" + isEqual);22         }23     }24 }

上面的代码运行的结果是true:

从这段代码来看,HttpContext其实是保存在System.Runtime.Remoting.Messaging.CallContext.HostContext这个属性中, System.Runtime.Remoting.Messaging.CallContext.HostContext在MSDN的解释是 获取或设置与当前线程相关联的主机上下文

我们在一个ASP.NET程序中,为什么可以到处访问HttpContext.Current呢?

因为ASP.NET会为每个请求分配一个线程(也是当前线程),这个线程会执行我们的代码来生成响应结果, 即使我们的代码散落在不同的地方(类库),线程仍然会执行它们, 所以我们可以在任何地方访问System.Web.HttpContext.Current获取到与当前请求相关的HttpContext对象, 这些代码是由同一个线程来执行,所以得到的HttpContext引用也就是我们期待的那个与请求相关的对象。

当前线程是什么意思?

我的理解是:
1. 当前线程是指与当前请求相关的线程。
2. 在ASP.NET程序中,有些线程并非总是与请求相关。

虽然在ASP.NET程序中,几乎所有的线程都应该是为响应请求而运行的,但是还有一些线程却不是为了响应请求而(产生)运行的,

例如:

1. 定时器的回调。
2. Cache的移除通知。
3. APM模式下异步完成回调。
4. 主动创建线程或者将任务交给线程池来执行。

5.异步任务Task

至于什么APM网上资料很多,这里我就不说明了

在这些情况下使用System.Web.HttpContext.Current获取HttpContext对象得到的结果都是null,因为处理他们的线程不是当前线程(为处理请求产生线程)

说的这里我们再回头看看本文开始写的(部分)代码:

1          /// 2         /// 获取文件绝对路径3         /// 4         /// 文件名称5         /// 
6 public string Test_3(string fileName)7 {8 return System.Web.HttpContext.Current.Server.MapPath("~/Log" + fileName);9 }

如果这段代码在那5种情况下运行,都会抛空指针异常,因为System.Web.HttpContext.Current得到是null。

为什么会得到null呢?

因为运行这段代码线程不是处理当前请求的当前线程

为什么其他地方得到又不是null呢?

因为ASP.NET程序在调用您的代码前,已经将HttpContext对象设置到前面所说的System.Runtime.Remoting.Messaging.CallContext.HostContext属性中。

HttpApplication有个内部方法OnThreadEnter(),ASP.NET在调用外部代码前会调用这个方法来切换HttpContext, 例如:每当执行管线的事件处理器之前,或者同步上下文(AspNetSynchronizationContext)执行回调时。 切换线程的CallContext.HostContext属性之后,我们的代码就可以访问到HttpContext引用。 注意:HttpContext的引用其实是保存在HttpApplication对象中。

 

这种情况下该如何获取文件的绝对路径呢?

我们可以访问System.Web.HttpRuntime.AppDomainAppPath获取程序的根路径,然后再拼接文件的相对路径即可

上面的代码得到的HttpContext对象是null,再调用MapPath来获取站点根目录,就必死无疑了!

所以在此建议大家在获取程序(站点)的根目录时尽量使用System.Web.HttpRuntime.AppDomainAppPath进行获取站点的根目录

 

那么在异步调用调用时访问HttpContext对象呢?

前面我还提到在APM模式下的异步完成回调时,访问HttpContext.Current也会返回null,那么此时该怎么办呢?

1. 在类型中添加一个字段来保存HttpContext的引用(异步开始前)。
2. 将HttpContext赋值给BeginXXX方法的最后一个参数(object state)

建议优先选择第二种方法,因为可以防止以后他人维护时数据成员被意外使用。

 

转载于:https://www.cnblogs.com/linJie1930906722/p/5708966.html

你可能感兴趣的文章
Appium for Android元素定位方法
查看>>
pfSense LAGG(链路聚合)设置
查看>>
教学思路SQL之入门习题《学生成绩》 七.存储过程基础知识
查看>>
createrepo 无法使用解决
查看>>
.net安全类库
查看>>
在Windows 2008 R2上部署SCCM 2007 R2
查看>>
tablespace backup模式一个没用的技术
查看>>
PostgreSQL安装
查看>>
七牛实时音视频云视频连线demo(web部分)
查看>>
Mysql 权限
查看>>
Spring事务管理(详解+实例)
查看>>
ubuntu apt-get install 出现无法定位软件包...
查看>>
centos7 下 基于docker搭建java/tomcat (方式一)
查看>>
全世界最好的编辑器VIM之Windows配置(gvim)[未测试]
查看>>
2018年你需要知道的13个JavaScript工具库
查看>>
当你点击按钮的时候如何设置其他按钮不可点击
查看>>
spring 高级装配
查看>>
【合集】parasoft Jtest 从安装到使用教程合集,收藏推荐!
查看>>
Python Pygame库的学习之路(1)
查看>>
信息安全与Linux系统
查看>>