学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(八)

6.线程安全

通过RouteTable的静态只读属性Routes表示的RouteCollection对象是针对整个应用的全局路由表。这个集合对象本身并不能提供线程安全的保证,所以同一个RouteCollection对象在多个线程中被同时操作就有可能造成意想不到的并发问题。为了解决这个问题,如下两个方法(GetReadLock和GetWriteLock)被定义在RouteCollection类型中,我们在对集合进行读取或者更新的时候可以分别调用他们获取读锁和写锁。

当执行GetReadLock方法的时候,只有在当前RouteCollection对象的写锁尚未被获取时才会将集合的读锁返回,否则会等待写锁的释放。当我们调用GetWriteLock方法视图获取某个RouteCollection对象的写锁的时候,针对该集合的写锁只有在没有任何线程拥有读锁和写锁的情况下才会返回,否则会等待所有锁的释放。也就是说线程安全状态下的RouteCollection对象可以被多个线程同时读取,但是不允许在被某个线程读取的同时被另一个线程更新。集合在某个时刻只能被一个线程更新,此时其他线程针对集合的读取和更新都是不允许的。

RouteCollection的GetReadLock和GetWriteLock方法的返回类型都是IDisposable接口,实际上返回值的类型分别是内嵌于RouteCollection中的两个私有类型(ReadLockDisposable和WriteLockDisposable),他们通过封装的RouterWriterLockSlim对象实现了读/写锁的功能。ReadLockDisposable和WriteLockDisposable实现了IDisposable接口,并在Dispose方法中完成对锁的释放,所以推荐的编程方式如下所示。

//读操作
using(IDisposable readLock = routeCollection.GetReadLock())
{
//读取RouteCollection
}
//写操作
using(IDisposable writeLock = routeCollection.GetWriterLock())
{
//更新RouteCollection
}

我们所说的路由注册本质就是创建相应的Route对象并将其添加到通过RouteTable的静态属性Routes表示的全局路由表中。照理说不论我们调用RouteCollection的Add方法或者MapPageRoute都需要预先获取集合的写锁,但是在一般情况下路由的注册发生在应用启动的时候(此时请求尚未抵达),能够确保集合对象此时只会被一个单一线程操作,所以在这种情况下我们无须调用GetWriteLock方法。值得一提的是,RouteCollection的两个方法GetRouteData和GetVirtualPath在对集合进行遍历之前已经调用了GetReadLock方法获得读锁,所以这两个方法本身就是线程安全的。

时间: 2024-11-06 14:20:47

学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(八)的相关文章

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(四)

Action的执行 作为Controller的基类ControllerBase,它的Execute方法主要作用在于执行目标Action方法.如果目标Action方法返回一个ActionResult对象,它还需要执行该对象来对当前请求予以响应.在ASP.NET MVC框架中,两者的执行是通过一个叫做ActionInvoker的对象来完成的. 1.ActionInvoker 我们同样为ActionInvoker定义了一个借口IActionInvoker.如下所示.该接口定义了唯一的方法InvokeA

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(一)

ASP.NET MVC是如何运行的 ASP.NET由于采用了管道式设计,所以具有很好的扩展性,整个ASP.NET MVC应用框架就是通过扩展ASP.NET实现的.通过上面对ASP.NET管道设计的介绍我们知道,ASP.NET 的扩展点主要体现在HttpModule和HttpHandler这两个核心组件之上,整个ASP.NET MVC框架就是通过自定义的HttpModule和HttpHandler建立起来的. 接下来我们通过自定义组件来模拟ASP.NET MVC的运行原理. 1.4.1建立在迷你版

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(五)

2.AreaRegistration的缓存 Area的注册(主要是基于Area的路由映射注册)通过具体的AreaRegistration来完成.在应用启动的时候,ASP.NET MVC会遍历通过调用BuildManager的静态方法GetReferencedAssemblies得到的程序集列表,并从中找到所有AreaRegistration类型.如果一个应用涉及太多的程序集,则这个过程可能会耗费很多时间.为了提高性能,ASP.NET MVC会对解析出来的所有AreaRegistration类型列

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(三)

Controller的激活 ASP.NET MVC的路由系统通过注册的路由表对当前HTTP请求实施路由解析,从而得到一个用于封装路由数据的RouteData对象,这个过程是通过自定义的UrlRoutingModule对HttpApplication的PostResolveRequestCache事件进行注册实现的.由于得到的RouteData对象中已经包含了目标Controller的名称,我们需要根据该名称激活对应的Controller对象. 1.MvcRouteHandler 通过前面的介绍我

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(五)

完整的流程 对于我们创建的这个迷你版的ASP.NET MVC框架来说,虽然很多细节被直接忽略掉,但是它基本上能够展现整个ASP.NET MVC框架的全貌,支持这个开发框架的核心对象可以说一个不少.接下来我们对通过这个模拟框架展现出来的ASP.NET MVC针对请求的处理流程作一个简单的概括. 由于UrlRoutingModule这个HttpModule被注册到Web应用中,所以对于每个抵达的请求来说,当代表当前应用的HttpApplication对象的PostResolveRequestCach

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC是如何运行的(二)

路由 对于一个ASP.NET MVC应用来说,针对HTTP请求的处理实现在目标Controller类型的某个Action,每个HTTP请求不在像ASP.NET Web Forms应用一样是针对一个物理文件,而是针对某个Controller的某个Action方法.目标Controller和Action的名称由HTTP请求的URL来决定,当ASP.NET MVC接收到抵达的请求后,其首要任务就是通过当前HTTP请求解析得到目标Controller和Action的名称,这个过程是通过ASP.NET M

学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(十一)

3.对现有物理文件的路由 在成功注册路由的情况下,如果我们按照传统的方式访问一个现存的物理文件,在请求地址满足某个Route的路由规则,ASP.NET是否还能正常实施路由呢?我们不妨通过实例来测试一下.为了让针对某个物理文件的访问地址也满足注册路由对象的路由模板采用的URL模式,我们需要按照如下的方式在进行路由注册时将表示约束的参数设置为Null. public class Global : System.Web.HttpApplication { protected void Applicat

学习ASP.NET MVC5框架揭秘笔记-ASP.NET路由(二)

实例演示:通过路由实现请求地址与.aspx页面的映射 我们创建一个简单的ASP.NET Web Forms应用,并采用一套独立于.aspx文件路径的URL来访问对应的Web页面,两者之间的映射通过路由来实现,我们依然沿用员工管理的场景. 首先我们将员工的所有信息(ID.姓名.性别.出生日期和所在部门)定义在如下所示的Employee类型中,然后定义一个EmployeeRepository类型来维护员工列表的数据.简单起见,员工列表通过静态字段employees表示.EmployeeReposit

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(二)

2.2.2 路由注册 ASP.NET MVC通过调用代表全局路由表的RouteCollection对象的扩展方法MapRoute进行路由注册.我们来进行一个简单的实例演示.我们依然沿用之前关于获取天气信息的路由模板,看看通过这种方式注册的Route对象针对匹配的请求将返回怎样一个RouteData对象. 我们创建一个空的ASP.NET Web程序,并手动添加"System.Web.Mvc.dll"和"System.Web.WebPages.Razor.dll"的引用

学习ASP.NET MVC5框架揭秘笔记-ASP.NET MVC路由(三)

2.2.3缺省URL参数 当通过VisualStudio的ASP.NET MVC项目模板创建一个Web应用后,它会为我们注册如下一个模板为"{controller}/{action}/{id}"的默认Route对象.3个路由模板均有相应额默认值.但是变量名为id的默认值为URLParameter.Optional.按照字面的意思,我们将其称为可缺省URL参数.那么将路由变量的默认值进行如此设置与设置一个具体的默认值有什么区别呢? routes.MapRoute( name: "