您的位置:首页 > 编程语言 > ASP

深入分析 ASP.NET Mvc 1.0 – 2. Controller.Execute(Request)-TempDataDictionary的Load与Save操作

2011-12-07 16:32 801 查看
Controller最终是通过调用ControllerBase类的Execute(RequestContext)方法来完成一个Action的创建与执行操作, 代码如下:

protected virtual void Execute(RequestContext requestContext) {
if (requestContext == null) {
throw new ArgumentNullException("requestContext");
}

Initialize(requestContext);
ExecuteCore();
}

代码分为两步:

Initialize(requestContext): 创建ControllerContext类的一个实例。
ExecuteCore(): 加载TempData, 创建及执行Action,处理Action返回的ActionResult , 保存TempData数据。

ExecuteCore()的代码如下:

protected override void ExecuteCore() {
TempData.Load(ControllerContext, TempDataProvider);

try {
string actionName = RouteData.GetRequiredString("action");
if (!ActionInvoker.InvokeAction(ControllerContext, actionName)) {
HandleUnknownAction(actionName);
}
}
finally {
TempData.Save(ControllerContext, TempDataProvider);
}
}

代码又分为三个部分:

TempData.Load(ControllerContext, TempDataProvider): 从HttpContextBase.Session中加载TempData数据

ActionInvoker.InvokeAction(ControllerContext, actionName): 创建,执行Action,并处理Action返回的ActionResult

TempData.Save(ControllerContext, TempDataProvider): 保存TempData

第1, 第3部分都是对TempData的操作,下面的文字将详细介绍这两个步骤。

1. TempData.Load(ControllerContext, TempDataProvider)

TempDataProvider: 就是SessionStateTempDataProvider,他是一个继承了ITempDataProvider接口的Session辅助类

TempDataProvider.Load的源码:

public void Load(ControllerContext controllerContext, ITempDataProvider tempDataProvider) {
IDictionary<string, object> providerDictionary = tempDataProvider.LoadTempData(controllerContext);
_data = (providerDictionary != null) ? new Dictionary<string, object>(providerDictionary, StringComparer.OrdinalIgnoreCase) :
new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
_initialKeys = new HashSet<string>(_data.Keys);
_modifiedKeys.Clear();
}

这里就是加载TempData的全部过程:

调用SessionStateTempDataProvider中的LoadTempData(…)方法并近回一个IDictionary<string, object>对象
初始化_data对象,如果providerDictionary不是null那么把providerDictionary中的数据存放到_data对象中,其实这个_data就是TempData用来保存数据的容器,
初始化_initialKeys对象并将_data中的全部Key值放入其中,它用来缓存已加载的Key值,如果对TempData中插入新数据时, _initialKeys对象中的数据不会有任务变化
清除_modifiedKeys中的值,用来保存TempData新的Key值,当有一个新的键值对插入到TempData中,这个key值会被保存到_modifiedKeys对象中

再来深入tempDataProvider.LoadTempData(controllerContext)方法,其实就是SessionStateTempDataProvider类中的LoadTempData(…)方法:

public virtual IDictionary<string, object> LoadTempData(ControllerContext controllerContext) {
HttpContextBase httpContext = controllerContext.HttpContext;

if (httpContext.Session == null) {
throw new InvalidOperationException(MvcResources.SessionStateTempDataProvider_SessionStateDisabled);
}

Dictionary<string, object> tempDataDictionary = httpContext.Session[TempDataSessionStateKey] as Dictionary<string, object>;

if (tempDataDictionary != null) {
// If we got it from Session, remove it so that no other request gets it
httpContext.Session.Remove(TempDataSessionStateKey);
return tempDataDictionary;
}
else {
return new Dictionary<string, object>(StringComparer.OrdinalIgnoreCase);
}
}

SessionStateTempDataProvider类中定义了一个常量,用它来作为保存在Session中TempData对象的Key。

internal const string TempDataSessionStateKey = "__ControllerTempData";

在LoadTempData(…)方法中, 首先在Session中查找与key相对应的Dictionary<string, object>对象,如果Session中存在这个对象,那么就清除Session中的这个键值对并返回找到的Dictionary<string, object>对象, 否则创建一个新的Dictionary<string, object>对象并返回。

这里的重点就是httpContext.Session.Remove(TempDataSessionStateKey) 方法,这也是TempData的特点,即同一个TempData只能被传递一次。当在Session中找到TempData后立即将它从Session中清除掉,下一次执行LoadTempData(…)方法时绝不会再找到同一个TempData。

再次说明:只要将一个键值对放入到TempData中,不论是同一个Controller中的不同Action,还是不同Controller中的不同Action都可以接收到这个TempData,但是仅此一次,当程序再次提交回服务端后是绝不可能再获得同样的值的TempData对象, 因为它已经在Session中被清除掉。

2. TempData.Save(ControllerContext, TempDataProvider)

这是整个Controller.Execute(…)生命周期中最后一个操作, 它的作用是保存新的TempData到HttpContext.Base.Session中。Save方法的具体代码:

public void Save(ControllerContext controllerContext, ITempDataProvider tempDataProvider) {
if (_modifiedKeys.Count > 0) {

// Apply change tracking.
foreach (string x in _initialKeys) {
if (!_modifiedKeys.Contains(x)) {
_data.Remove(x);
}
}

// Store the dictionary
tempDataProvider.SaveTempData(controllerContext, _data);
}
}


上面提到过_initialKeys和_modifiedKeys的重要性,它们的作用在这里将体现出来,

_initialKeys: 保存在Action执行之前TempData中的全部Key值,如果在action执行过程中有新的键值对插入到TempData中, 对_initialKeys对象中的值不会有任何影响。
_modifiedKeys: 在Action执行过程中, 如果有新的键值对插入到TempData中,这个新的Key会插入到_modifiedKeys对象中。

Save方法中首先判断_modifiedKeys对象中元素的数量,是否有新的键值对被插入到TempData中,foreach和下面的if语句是从TempData中删除无用的键值对数据,最后调用 tempDataProvider.SaveTempData(controllerContext, _data)方法,将TempData保存在HttpContextBase.Session中。

注意:在加载TempData过程中会将上次定义的TempData中数据放入到_data中,这里的foreach, if语句是去掉不再需要传递的键值对,确保TempData中仅保留本次Action执行过程中产生的键值对数据。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐