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

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);
            }
        }

代码又分为三个部分:

  1. TempData.Load(ControllerContext, TempDataProvider): 从HttpContextBase.Session中加载TempData数据
  2. ActionInvoker.InvokeAction(ControllerContext, actionName): 创建,执行Action,并处理Action返回的ActionResult
  3. 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的全部过程:

  1. 调用SessionStateTempDataProvider中的LoadTempData(…)方法并近回一个IDictionary<string, object>对象
  2. 初始化_data对象,如果providerDictionary不是null那么把providerDictionary中的数据存放到_data对象中,其实这个_data就是TempData用来保存数据的容器,
  3. 初始化_initialKeys对象并将_data中的全部Key值放入其中,它用来缓存已加载的Key值,如果对TempData中插入新数据时, _initialKeys对象中的数据不会有任务变化
  4. 清除_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执行过程中产生的键值对数据。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值