zoukankan      html  css  js  c++  java
  • mvc源码解读(16)数据绑定组件ModelBinder之ModelBinderProviders

        在上一篇中我们讲到了应用在Action方法参数上述的ModelBinder,源码具体分析到:

     private IModelBinder GetModelBinder(ParameterDescriptor parameterDescriptor) {
                return parameterDescriptor.BindingInfo.Binder ?? Binders.GetBinder(parameterDescriptor.ParameterType);
            }

    我们已经讲了ReflectedParameterBindingInfo的Binder属性,假如它为空的话,则根据参数类型来获取ModelBinder,GetBinder方法的具体实现是在ModelBinderDictionary类里面:

     public IModelBinder GetBinder(Type modelType) {
                return GetBinder(modelType, true /* fallbackToDefault */);
            }

    这个方法最终调用的又是GetBinder的一个重载方法,这个方法的第二个参数是一个bool类型,具体实现如下:

    public virtual IModelBinder GetBinder(Type modelType, bool fallbackToDefault) {
                if (modelType == null) {
                    throw new ArgumentNullException("modelType");
                }
                //fallbackToDefault表示是否启用默认的DefaultModelBinder.
                return GetBinder(modelType, (fallbackToDefault) ? DefaultBinder : null);
            }

     我们可以发现GetBinder的第二个参数fallbackToDefault为true的时候表示启用mvc默认的DefaultModelBinder。该方法里面调用的又是GetBinder的另外一个重载方法,该方法也是在ModelBinderDictionary类里面定义的,方法具体实现如下:

    private IModelBinder GetBinder(Type modelType, IModelBinder fallbackBinder) {

                 // Try to look up a binder for this type. We use this order of precedence:            

                 // 1. Binder returned from provider            

                // 2. Binder registered in the global table            

               // 3. Binder attribute defined on the type            

              // 4. Supplied fallback binder

                IModelBinder binder = _modelBinderProviders.GetBinder(modelType);            

                if (binder != null) {return binder;}

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

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

                return binder ?? fallbackBinder;        

    }

    上面列举了获取ModelBinder的四种优先顺序,我们先来看第一个优先选择的ModelBinder,是从provider里面获取的ModelBinder,具体实现是在类ModelBinderProviderCollection里面:

     public IModelBinder GetBinder(Type modelType) {

                if (modelType == null) {throw new ArgumentNullException("modelType");}

                var modelBinders = from providers in CombinedItems                               

                                             let modelBinder = providers.GetBinder(modelType)                               

                                             where modelBinder != null                               

                                             select modelBinder;

                return modelBinders.FirstOrDefault();        

    }

    对于类ModelBinderProviderCollection顾名思义我们理解为是ModelBinderProvider的一个集合,里面包括了当前应用的ModelBinderProvider列表,上面的红色代码中GetBinder方法实现的是接口IModelBinderProvider里面的GetBinder,该方法具体定义如下:

     public interface IModelBinderProvider {
            IModelBinder GetBinder(Type modelType);
        }

    但是mvc中并没有提供实现该接口的类,我在看源码的时候一直卡在这里想不通,知道看到了ModelBinderProviders这个类,我才知道了它的用意,这个先卖个关子,我们直接看ModelBinderProviders类里面的成员:

     public static class ModelBinderProviders {

            private readonly static ModelBinderProviderCollection _binderProviders = new ModelBinderProviderCollection {};

            public static ModelBinderProviderCollection BinderProviders {

                           get {return _binderProviders;}        

            }    

    }

    BinderProviders属性就是对ModelBinderProviderCollection的封装,而我们可以看到ModelBinderProviderCollection是继承自 Collection<IModelBinderProvider>,定义如下:

    public class ModelBinderProviderCollection : Collection<IModelBinderProvider>

    那既然mvc没有提供实现IModelBinderProvider的类,那ModelBinderProviders的属性BinderProviders表示的ModelBinderProvider列表什么时候被调用的呢?看过源码我们发现最终是被ModelBinderDictionary来使用,我们来看ModelBinderDictionary的一些成员:

    private IModelBinder _defaultBinder;
    private readonly Dictionary<Type, IModelBinder> _innerDictionary = new Dictionary<Type, IModelBinder>();
    private ModelBinderProviderCollection _modelBinderProviders;

    里面包含了ModelBinderProviderCollection类的实例,那这个实例是什么时候被初始化的呢?初始化之前我们知道这个集合里里面肯定是没有数据,因此我们在mvc应用程序的全局文件Global.asax里面的Application_Start()里面进行初始化的工作。如下所示:

       ModelBinderProviders.BinderProviders.Add(******);

    这样的话从provider就获取到了ModelBinder。

  • 相关阅读:
    论在Repository中使用EF框架
    SQL字符串函数
    网站可用性测试及优化指南-随笔2
    对线上系统维护工作的总结与思考
    SQL 判断字段中指定字符出现的次数
    SQL SERVER 的 INFORMATION_SCHEMA 的使用
    查看SQL语句执行时间
    Bootstrap框架中的字形图标的理解
    字符串编码、Base64字符串 互转
    根据端口号查应用程序pid
  • 原文地址:https://www.cnblogs.com/ghhlyy/p/2941171.html
Copyright © 2011-2022 走看看