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()方法的源代码:

Code
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()被调用,

Code
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的定义:

Code
protected   internal  ModelBinderDictionary Binders {
            
get  {
                
if  (_binders  ==   null ) {
                    _binders 
=  ModelBinders.Binders;
                }
                
return  _binders;
            }
            
set  {
                _binders 
=  value;
            }
        }


这里,Binders引用的是ModelBinders.Binders(注意到ModelBinders是个静态类),这时候ModelBinderDictionary要开始发挥作用了,来看ModelBinderDictionary类中的GetBind()方法:

Code
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。

Copyright 作者: Tristan G
本文版权归作者和博客园共有,欢迎转载 :-)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值