4

我有以下一组枚举:

[Flags]
public enum Categories : uint
{
    A = (1 << 0),
    B = (1 << 1),
    B1 = B | (1 << 16),
    B2 = B | (1 << 17),
    B3 = B | (1 << 18),
    B4 = B | (1 << 19),
    B5 = B | (1 << 20),
    C = (1 << 2),
    C1 = C | (1 << 21),
    D = (1 << 3),
    D1 = D | (1 << 22),
    D2 = D | (1 << 23),
    E = (1 << 4),
    F = (1 << 5),
    F1 = F | (1 << 23),
    F2 = F | (1 << 24),
    F3 = F | (1 << 25),
    F4 = F | (1 << 26),
    F5 = F | (1 << 27),
    G = (1 << 6),
    H = (1 << 7),
    H1 = H | (1 << 28),
}

这个想法是枚举表示一个层次结构,其中子枚举暗示其父级,并且可以应用任意数量的标志。

我看到的问题是,在调试过程中,所有子枚举都没有被表示为名称或名称集。IE,Categories.F=“F”但Categories.F2= 16777248。我希望Categories.F2=“F,F2”或至少“F2”

我怎样才能让我的枚举仍然被识别为标志?有没有更好的方法来完成我想要做的事情?

4

2 回答 2

3

很奇怪,调试器中的值与ToString值不同。根据文档,两者应该匹配(因为Enum类型确实会覆盖ToString)。

如果 C# 对象有一个覆盖ToString(),调试器将调用覆盖并显示其结果而不是标准的{<typeName>}.

显然这不适用于枚举。我最好的猜测是调试器正在尝试对枚举类型进行一些特殊的、未记录的处理。添加DebuggerDisplayAttribute显然通过覆盖此行为解决了该问题。

[DebuggerDisplay("{ToString()}")]
[Flags]
public enum Categories : uint
{
    ...
}

Categories.F2.ToString() = "F, F2"

C# 不会为你做那个魔法,因为F2在枚举中已经有了它自己的名字。您可以像这样手动标记各个成员:

public enum Categories
{
    [Description("F, F2")]
    F2 = F | (1 << 24),
}

然后编写代码转换为描述。

public static string ToDescription(this Categories c)
{
    var field = typeof(Categories).GetField(c.ToString());
    if (field != null)
    {
        return field.GetCustomAttributes().Cast<DescriptionAttribute>().First().Description;
    }
}
...
Categories.F2.ToDescription() == "F, F2";

或者你可以做一些魔法来自己生成这个:

public static string ToDescription(this Categories c)
{
    var categoryNames =
        from v in Enum.GetValues(typeof(Categories)).Cast<Category>()
        where v & c == c
        orderby v
        select v.ToString();
    return String.Join(", ", categoryNames);
}

不幸的是,扩展方法不能与 一起使用DebuggerDisplayAttribute,但你可以使用DebuggerTypeAttributeYMMV 但你可以试试这个:

[DebuggerType("CategoryDebugView")]
[Flags]
public enum Categories : uint
{
    ...
}

internal class CategoryDebugView
{
    private Category value;

    public CategoryDebugView(Category value)
    {
        this.value = value;
    }

    public override string ToString()
    {
        var categoryNames =
            from v in Enum.GetValues(typeof(Categories)).Cast<Category>()
            where v & c == c
            orderby v
            select v.ToString();
        return String.Join(", ", categoryNames);
    }
}
于 2013-04-06T06:31:09.573 回答
1

你可以通过一些工作来完成你所要求的。我创建了一些扩展方法Categories,用于HasFlag()确定枚举值是否具有特定的父级,然后调用ToString()它们并连接结果。

public static class CategoriesExtensionMethods
{
    public static Categories GetParentCategory(this Categories category)
    {
        Categories[] parents = 
        {
            Categories.A,
            Categories.B,
            Categories.C,
            Categories.D,
            Categories.E,
            Categories.F,
            Categories.G,
            Categories.H,
        };

        Categories? parent = parents.SingleOrDefault(e => category.HasFlag(e));
        if (parent != null)
            return (Categories)parent;
        return Categories.None;
    }

    public static string ToStringWithParent(this Categories category)
    {
        var parent = GetParentCategory(category);
        if (parent == Categories.None)
            return category.ToString();
        return string.Format("{0} | {1}", parent.ToString(), category.ToString());
    }
}

然后我们可以这样使用它:

var f1 = Categories.F1;

var f1ParentString = f1.ToStringWithParent();
// f1ParentString = "F | F1"

var f = Categories.F;
var fParentString = f.GetParentCategory();
// fParentString = "F"

更新

GetParentCategory()如果您不想指定所有父母,这是一种更漂亮的实施方式。

public static Categories GetParentCategory(this Categories category)
{
    var values = Enum.GetValues(typeof(Categories)).Cast<Categories>();
    var parent = values.SingleOrDefault(e => category.HasFlag(e) && e != Categories.None && category != e);
    if (parent != Categories.None)
        return (Categories)parent;
    return Categories.None;
}
于 2013-04-06T06:46:53.233 回答