23

我很想知道是否可以覆盖已在模型上设置的 [Required] 属性。我敢肯定这个问题有一个简单的解决方案,任何接受者?

4

5 回答 5

25

取决于你在做什么。如果您正在使用子类,使用具有 Required 属性的模型作为基础,您可以这样做:

使用关键字重新定义属性new,而不是覆盖它。

public class BaseModel
{
    [Required]
    public string RequiredProperty { get; set; }
}


public class DerivativeModel : BaseModel
{
    new public string RequiredProperty { get; set; }

}

如果您只是想绑定或验证模型,但跳过控制器中的 Required 属性,您可以执行以下操作:

public ActionResult SomeAction()
{
     var model = new BaseModel();

     if (TryUpdateModel(model, null, null, new[] { "RequiredProperty" })) // fourth parameter is an array of properties (by name) that are excluded
     {
          // updated and validated correctly!
          return View(model);
     }
     // failed validation
     return View(model);
}
于 2012-01-18T00:58:42.737 回答
18

@HackedByChinese 方法很好,但它包含一个问题

public class BaseModel
{
    [Required]
    public string RequiredProperty { get; set; }
}

public class DerivativeModel : BaseModel
{
    new public string RequiredProperty { get; set; }
}

ModelState即使您DerivativeModel在表单上使用此代码也会给您一个验证错误,override也不起作用,因此您无法Required通过覆盖或更新属性来删除属性,所以我找到了某种解决方法

public class BaseModel
{
    public virtual string RequiredProperty { get; set; }
}

public class DerivativeModel : BaseModel
{
    [Required]
    public override string RequiredProperty { get; set; }
}

public class DerivativeModel2 : BaseModel
{
    [Range(1, 10)]
    public override string RequiredProperty { get; set; }
}

我有一个没有验证属性和派生类的基本模型

于 2015-07-16T07:54:20.233 回答
7

您可以使用自定义验证属性(它可能派生自RequiredAttribute):

 public class RequiredExAttribute : RequiredAttribute
    {
        public bool UseRequiredAttribute { get; protected set; }
        public RequiredExAttribute(bool IsRequired)
        {
            UseRequiredAttribute = IsRequired;
        }
        public override bool IsValid(object value)
        {
            if (UseRequiredAttribute)
                return base.IsValid(value);
            else
            {
                return true;
            }
        }

        public override bool RequiresValidationContext
        {
            get
            {
                return UseRequiredAttribute;
            }
        }
    }

    public class RequiredExAttributeAdapter : RequiredAttributeAdapter
    {
        public RequiredExAttributeAdapter(ModelMetadata metadata, ControllerContext context, RequiredExAttribute attribute)
            : base(metadata, context, attribute) { }

        public override IEnumerable<ModelClientValidationRule> GetClientValidationRules()
        {
            if (((RequiredExAttribute)Attribute).UseRequiredAttribute)// required -> return normal required rules
                return base.GetClientValidationRules();
            else// not required -> return empty rules list
                return new List<ModelClientValidationRule>();
        }
    }

然后Application_Start使用此代码行对其进行注册:

 DataAnnotationsModelValidatorProvider.RegisterAdapter(typeof(RequiredExAttribute), typeof(RequiredExAttributeAdapter));
于 2014-06-17T08:40:35.977 回答
2

是的,可以使用MetadataType类,例如:

[MetadataType(typeof(Base.Metadata))]
public class Base
{    
    public string RequiredProperty { get; set; }

    public class Metadata
    {
        [Required]
        public string RequiredProperty { get; set; }
    }
}

[MetadataType(typeof(Derived.Metadata))]
public class Derived : Base 
{
    public new class Metadata
    {
    }
}

并测试它:

var type = typeof(Derived);

var metadataType = typeof(Derived.Metadata);

var provider = new AssociatedMetadataTypeTypeDescriptionProvider(type, metadataType);

TypeDescriptor.AddProviderTransparent(provider, type);

var instance = new Derived();

var results = new List<ValidationResult>();

Validator.TryValidateObject(instance,
    new ValidationContext(instance),
    results,
    true);

Debug.Assert(results.Count == 0);
于 2017-03-13T15:03:35.887 回答
1

我尝试了 Mahmoud 的答案,但如果没有一些更改,它对我不起作用。将此添加为答案,以便我可以提供代码以防万一它对其他人有所帮助,但完全归功于 Mahmoud Hboubati - 我已赞成您的答案。

在我的情况下,我有一个带有 DbGeography 属性的基本 DTO 类,这是 MVC 项目所需的,该项目使用自定义 EditorTemplate 和 DisplayTemplate 作为 DbGeography 类型。但是为了将模型发布到 Web API 控制器,我希望将纬度/经度字段添加到该 DTO 的子类中,这将用于创建和设置 DbGeography 类的实例以设置 DbGeography 属性的值。问题是,我不能只在子类上设置不需要的 DbGeography 属性。

当使用 Mahmoud 的方法在构造函数中传递布尔值时,它似乎从未覆盖我的默认值。这可能是因为我正在使用 Web API 并使用工厂方法注册属性,如下所示(在 Global.asax.cs Application_Start 方法中):

DataAnnotationsModelValidationFactory factory = (p, a) => new DataAnnotationsModelValidator(
    new List<ModelValidatorProvider>(), new RequiredExAttribute()
);

DataAnnotationsModelValidatorProvider provider = new DataAnnotationsModelValidatorProvider();
provider.RegisterAdapterFactory(typeof(RequiredExAttribute), factory);

我不得不将属性类更改为:

using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Web.Mvc;
...
public class RequiredExAttribute : RequiredAttribute
{
    public bool IsRequired { get; set; }

    public override bool IsValid(object value)
    {
        if (IsRequired)
            return base.IsValid(value);
        else
        {
            return true;
        }
    }

    public override bool RequiresValidationContext
    {
        get
        {
            return IsRequired;
        }
    }
}

public class RequiredExAttributeAdapter : RequiredAttributeAdapter
{
    public RequiredExAttributeAdapter(ModelMetadata metadata, ControllerContext context, RequiredExAttribute attribute)
        : base(metadata, context, attribute) { }

    public override IEnumerable<ModelClientValidationRule> GetClientValidationRules()
    {
        if (((RequiredExAttribute)Attribute).IsRequired)// required -> return normal required rules
            return base.GetClientValidationRules();
        else// not required -> return empty rules list
            return new List<ModelClientValidationRule>();
    }
}

基类:

[RequiredEx(IsRequired = true)]
public virtual DbGeography Location { get; set; }

子类:

[RequiredEx(IsRequired = false)]
public override DbGeography Location { get; set; }

[Required]
public decimal Latitude { get; set; }

[Required]
public decimal Longitude { get; set; }

注意,我使用与 Mahmoud 上面相同的方法在我的 MVC 项目中注册属性:

DataAnnotationsModelValidatorProvider.RegisterAdapter(typeof(RequiredExAttribute), typeof(RequiredExAttributeAdapter));
于 2016-11-10T18:09:38.457 回答