asp.net-mvc-2 – ASP.NET MVC2 ModelMetadataProviders:重写CreateMetadata()和GetMetadataForProperty()之间有什么区别?

前端之家收集整理的这篇文章主要介绍了asp.net-mvc-2 – ASP.NET MVC2 ModelMetadataProviders:重写CreateMetadata()和GetMetadataForProperty()之间有什么区别?前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
MetadataProviders开始,我很想探索框架的扩展点.我目前已经成功实施了 populating ModelMetadata.IsRequired property using RequiredAttribute,但我似乎无法找到它们之间的区别
重写CreateMetadata()或GetMetadataForProperty(),因为两个选项似乎都有效.

通常,我见过的示例会覆盖CreateMetadata().

>使用这两种选择的利弊是什么?
>是否存在其中一个是首选方案的情况?

作为额外的:是否有任何好的资源(博客,书籍)可以从这个扩展点学习?

解决方法

GetMetadataForProperty()在类ModelMetadataProvider上声明.

AssociatedMetadataProvider派生自ModelMetadataProvider. CreateMetadata()在AssociatedMetadataProvider上声明.在您提供的链接中重写的DataAnnotationsMetadataProvider派生自AssociatedMetadataProvider.

MVC框架调用ModelMetadataProvider的GetMetadataForProperty()方法.

覆盖CreateMetadata()的原因是因为AssociatedModelMetadataProvider的默认GetMetadataForProperty()实现了对CreateMetadata()的调用.它看起来像这样:

public override ModelMetadata GetMetadataForProperty(Func<object> modelAccessor,Type containerType,string propertyName)
{
   if (containerType == null)
   {
       throw new ArgumentNullException("containerType");
   }
   if (string.IsNullOrEmpty(propertyName))
   {
       throw new ArgumentException(MvcResources.Common_NullOrEmpty,"propertyName");
   }
   PropertyDescriptor propertyDescriptor = this.GetTypeDescriptor(containerType).GetProperties().Find(propertyName,true);
   if (propertyDescriptor == null)
   {
       throw new ArgumentException(string.Format(CultureInfo.CurrentCulture,MvcResources.Common_PropertyNotFound,new object[] { containerType.FullName,propertyName   }));
   }
return this.GetMetadataForProperty(modelAccessor,containerType,propertyDescriptor);

}

protected virtual ModelMetadata GetMetadataForProperty(Func<object> modelAccessor,PropertyDescriptor propertyDescriptor) 
{
   IEnumerable<Attribute> attributes = this.FilterAttributes(containerType,propertyDescriptor,propertyDescriptor.Attributes.Cast<Attribute>());
   return this.CreateMetadata(attributes,modelAccessor,propertyDescriptor.PropertyType,propertyDescriptor.Name);
}

如果您正在将AssociatedMetadataProvider子类化为您提供的链接,则首选的可扩展性点是CreateMetadata方法,因为AssociatedMetadataProvider.GetMetadataForProperty()方法预先验证CreateMetadata()方法的合约.这样,您就知道如果您的CreateMetadata()方法中存在错误,您已经知道错误的来源在您的方法中,而不是在传递给它的参数中.

另外,这里是FilterAttributes()方法的源代码,以防你想知道:

protected virtual IEnumerable<Attribute> FilterAttributes(Type containerType,PropertyDescriptor propertyDescriptor,IEnumerable<Attribute> attributes)
{
if (!typeof(ViewPage).IsAssignableFrom(containerType) && !typeof(ViewUserControl).IsAssignableFrom(containerType))
    {
        return attributes;
    }
    return attributes.Where<Attribute>(delegate (Attribute a) {
        return !(a is ReadOnlyAttribute);
    });
}

猜你在找的asp.Net相关文章