当通过反射检索到值时,枚举失败==

问题描述 投票:1回答:1

我正在尝试为MVC 6实现自定义jQuery Unobtrusive Validation属性。下面是一个属性的IsValid()实现,它查看类中的相邻属性并将其与编译时常量进行比较。

protected override ValidationResult IsValid(object value, ValidationContext validationContext)
{
    ValidationResult result = ValidationResult.Success;

    // Simplest case, there is something in this field
    if (!string.IsNullOrEmpty(value?.ToString()))
        return result;

    // Check relative field
    try
    {
        // Grab the property referenced via reflection
        var relativeProperty = validationContext.ObjectType.GetProperty(this._relativePropertyName);

        // Compare the runtime value of that property to the Attribute initialized value
        if (relativeProperty.GetValue(validationContext.ObjectInstance) == this._relativeValue)
        {
            // Fail if those 2 values are equal
            result = new ValidationResult(this.ErrorMessageString); 
        }

    }
    catch (Exception)
    {
        result = new ValidationResult(this.ErrorMessageString);
    }

    return result;
}

在大多数情况下,这完全符合预期。我唯一的问题是当引用的属性及其值是Enum的衍生物时(例如,让我们假装我们的枚举是Result)。实现此属性在类中如下所示:

public class TestObject
{
    public Result TestResult { get; set; }
    [IsEqual(nameof(TestResult), Result.Pass)]
    public string PassComment { get; set; }
}

当调试器到达行if(relativeProperty.GetValue(validationContext.ObjectInstance) == this._relativeValue)时,即使relativeProperty.GetValue(validationContext.ObjectInstance)解析为Result.Pass,它也会失败。使用立即窗口,relativeProperty.GetValue(validationContext.ObjectInstance) = Result.Pass以及this._relativeValue = Result.Pass

小方注意,在两个值上调用.GetType()也是相同的。

我假设这与拳击有关,但无法确定它。提前致谢。

c# asp.net asp.net-mvc unobtrusive-validation
1个回答
5
投票

不要使用==来评估非强类型事物的相等性。考虑以下:

void Main()
{
    Console.WriteLine(((object)Result.Pass) == (object)Result.Pass);
    // False

    Console.WriteLine(((object)Result.Pass).Equals((object)Result.Pass));
    // True

    Console.WriteLine(object.Equals((object)Result.Pass,(object)Result.Pass));
    // True
}

public enum Result{
    Pass, Fail
}

在C#中,可以覆盖==运算符,以便在比较某些类型的对象时使语法更容易,如果您尝试使用==这两种不兼容的类型,则可以提供编译时错误。有些人认为它看起来更好。

但是,当值向下转换为objects时,object == object运算符执行Object.ReferenceEquals()检查。由于枚举是值类型,因此必须将它们“装箱”到新对象中才能被转换为objects,并且这些新对象将存在于不同的内存位置。因此,ReferenceEquals()将是假的。

相反,使用object.Equals()方法,该方法传递给单个类的.Equals()覆盖。在这种情况下,枚举类型的.Equals()覆盖将检查该值是否可以转换为Result,以及结果值是否与this one相同。

object.Equals(value1, value2)优于value1.Equals(value2)的优点是,如果value1为null,则不会得到null异常。

© www.soinside.com 2019 - 2024. All rights reserved.