Asp.Net Mvc: Model Binding 机制分析

环境:

Windows 2008, VS 2008 SP1, Asp.Net Mvc RC1

------------------------------------------------------------------------------

本文通过分析Action Invoking的请求过程片段,探索当前Asp.Net Mvc中的Model Binding是如何实现的。

请求过程片段:

在请求的Action被调用之前,ControllerActionInvoker.InvokeAction()方法被调用,在这个方法里面,有一个ReflectedActionDescriptor的实例会被构建,这个实例包含Action的描述信息。

接着,ControllerActionInvoker.GetParameterValues()方法被调用,通过传入的之前创建的ReflectedActionDescriptor实例,获取Action参数列表(对应的ParameterDescriptor的实例分别被创建),进而遍历各参数,尝试获取参数的值,在遍历的循环里面,ControllerActionInvoker.GetParameterValue()方法被调用。

以下就是ControllerActionInvoker.GetParameterValue()方法的源代码:

protected virtual object GetParameterValue(ControllerContext controllerContext, ParameterDescriptor parameterDescriptor) {
            // collect all of the necessary binding properties
            Type parameterType = parameterDescriptor.ParameterType;
            IModelBinder binder = GetModelBinder(parameterDescriptor);
            IDictionary<string, ValueProviderResult> valueProvider = controllerContext.Controller.ValueProvider;
            string parameterName = parameterDescriptor.BindingInfo.Prefix ?? parameterDescriptor.ParameterName;
            Predicate<string> propertyFilter = GetPropertyFilter(parameterDescriptor);

            // finally, call into the binder
            ModelBindingContext bindingContext = new ModelBindingContext() {
                FallbackToEmptyPrefix = (parameterDescriptor.BindingInfo.Prefix == null), // only fall back if prefix not specified
                ModelName = parameterName,
                ModelState = controllerContext.Controller.ViewData.ModelState,
                ModelType = parameterType,
                PropertyFilter = propertyFilter,
                ValueProvider = valueProvider
            };
            object result = binder.BindModel(controllerContext, bindingContext);
            return result;
        }

这个方法做的事情就是,通过ParameterDescriptor的参数描述信息,获取一个与该参数关联的IModelBinder,进而调用IModelBinder.BindModel()方法,得到参数的值。

因此,这里有两个过程:

1) 确定IModelBinder

ControllerActionInvoker.GetModelBinder()被调用,

private IModelBinder GetModelBinder(ParameterDescriptor parameterDescriptor) {
            // look on the parameter itself, then look in the global table
            return parameterDescriptor.BindingInfo.Binder ?? Binders.GetBinder(parameterDescriptor.ParameterType);
        }

该方法首先尝试从ParameterDescriptor中获取IModelBinder(读取参数的CustomModelBinderAttribute),如果我们没有给参数指定Attribute,则为空,则从Binders中获取,请看这个Binders的定义:
protected internal ModelBinderDictionary Binders {
            get {
                if (_binders == null) {
                    _binders = ModelBinders.Binders;
                }
                return _binders;
            }
            set {
                _binders = value;
            }
        }

这里,Binders引用的是ModelBinders.Binders(注意到ModelBinders是个静态类),这时候ModelBinderDictionary要开始发挥作用了,来看ModelBinderDictionary类中的GetBind()方法
public IModelBinder GetBinder(Type modelType) {
            return GetBinder(modelType, true /* fallbackToDefault */);
        }

        public virtual IModelBinder GetBinder(Type modelType, bool fallbackToDefault) {
            if (modelType == null) {
                throw new ArgumentNullException("modelType");
            }

            return GetBinder(modelType, (fallbackToDefault) ? DefaultBinder : null);
        }

        private IModelBinder GetBinder(Type modelType, IModelBinder fallbackBinder) {
            // Try to look up a binder for this type. We use this order of precedence:
            // 1. Binder registered in the global table
            // 2. Binder attribute defined on the type
            // 3. Supplied fallback binder

            IModelBinder binder;
            if (_innerDictionary.TryGetValue(modelType, out binder)) {
                return binder;
            }

            binder = ModelBinders.GetBinderFromAttributes(modelType,
                () => String.Format(CultureInfo.CurrentUICulture, MvcResources.ModelBinderDictionary_MultipleAttributes, modelType.FullName));

            return binder ?? fallbackBinder;
        }

这是该方法的3个重载,按顺序被调用,当第三个重载方法被调用时,它首先会尝试从ModelBinderDictionary的_innerDictionary变量中取modelType对应的IModelBinder(前面提到ModelBinders是个静态类,因此我们就有机会添加类型和自定义IModelBinder的映射到这个字典里面),否则再次尝试从Attribute中获取,如果都找不到,那么就返回DefaultBinder,这里DefaultBinder是DefaultModelBinder的一个实例。最终,与参数对应的IModelBinder被宣告确定,接下来就是通过IModelBinder.BindMode()方法来给参数赋值了。

2) 取值赋值

接下来的事情,就是对应的IModelBinder发挥作用,结合controllerContext.Controller.ValueProvider来取值,并对参数赋值。

....

然后被请求的Action被正式Executed。

总结:

本文仅仅分析了一个很小的片段,关于更多相关的东西,将在以后的文章中继续探索。

以下列出相关的几个类,大家有兴趣可以看看:

CustomModelBinderAttribute, ModelBinderAttribute, BindAttribute, IModelBinder, DefaultModelBinder, ModelBinderDictionary, ModelBinders, ModelBindingContext, 以及几个xxxDescriptor。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值