23

我已经编写了代码来TryParse按值或按名称进行枚举,如下所示。如何扩展此代码以包含解析具有Flags属性的枚举?

    public static bool TryParse<T>(this T enum_type, object value, out T result) 
                where T : struct
            {
                return enum_type.TryParse<T>(value, true, out result);
            }



 public static bool TryParse<T>(this T enum_type, 
object value, bool ignoreCase, out T result)
        where T : struct
    {
        result = default(T);
        var is_converted = false;

        var is_valid_value_for_conversion = new Func<T, object, bool, bool>[]{
            (e, v, i) => e.GetType().IsEnum,
            (e, v, i) => v != null,
            (e, v, i) => Enum.GetNames(e.GetType()).Any(n => String.Compare(n, v.ToString(), i) == 0) || Enum.IsDefined(e.GetType(), v)
        };

        if(is_valid_value_for_conversion.All(rule => rule(enum_type, value, ignoreCase))){
            result = (T)Enum.Parse(typeof(T), value.ToString(), ignoreCase);
            is_converted = true;
        }

        return is_converted;
    }

目前,此代码适用于以下枚举:

enum SomeEnum{ A, B, C } 
// can parse either by 'A' or 'a'

enum SomeEnum1 : int { A = 1, B = 2, C = 3 } 
// can parse either by 'A' or 'a' or 1 or "1"

不适用于:

[Flags]
enum SomeEnum2 { A = 1, B = 2, C = 4 } // can parse either by 'A' or 'a'
// cannot parse for A|B

谢谢!

4

3 回答 3

45

标志枚举是,使用 .Net 约定而不是 .Net 约定编写的|。Enum.Parse() 在使用 ',' 字符串时效果很好:

[Flags]
public enum Flags
{
    A = 1,
    B = 2,
    C = 4,
    D = 8,
}

var enumString =  (Flags.A | Flags.B | Flags.C).ToString();
Console.WriteLine(enumString); // Outputs: A, B, C
Flags f = (Flags)Enum.Parse(typeof(Flags),  enumString);
Console.WriteLine(f); // Outputs: A, B, C
于 2010-04-30T14:57:34.757 回答
22

从 .NET 4 开始,有一个Enum.TryParse<T>方法。它支持开箱即用的标志枚举:

string x = (SomeEnum2.A | SomeEnum2.B).ToString();  //  x == "A, B"
SomeEnum2 y;
bool success = Enum.TryParse<SomeEnum2>(x, out y);  //  y == A|B
于 2010-04-30T14:57:22.493 回答
0

@Pop 的回答给了我一个线索,我修改了代码中的规则检查,如下所示:

var is_valid_value_for_conversion = new Func<T, object, bool, bool>[]
{
    (e, v, i) => e.GetType().IsEnum,
    (e, v, i) => value != null,
    (e, v, i) => Enum.GetNames(e.GetType()).Any(
                n => String.Compare(n, v.ToString(), i) == 0 
                || (v.ToString().Contains(",") && v.ToString().ToLower().Contains(n.ToLower()))) 
                || Enum.IsDefined(e.GetType(), v)
};

其余的保持不变,它对我有用

HTH 别人

于 2010-04-30T15:31:06.453 回答