3

我有一个看起来像这样的枚举:

enum foo{
a=0,
b=1,
c=2,
d=4
}

建立标志/位掩码很好,但是否可以做类似的事情

int i = 3;
var bar =  Enum.Split(foo,i);

导致类似

bar = foo[]{a, b,c};

谢谢。

4

6 回答 6

7

尝试以下

public static IEnumerable<T> Split<T>(int value) {
  foreach (object cur in Enum.GetValues(typeof(T))) {
    var number = (int)(object)(T)cur;
    if (0 != (number & value)) {
      yield return (T)cur;
    }
  }
}

有了这个你现在可以写

int i = 3;
IEnumerable<foo> e = Split<foo>(i);

注意:这仅适用于enum派生自int(默认设置)的值。它也不是完全类型安全的,因为T不能仅限于enum值(尽管对此无所谓)

于 2011-03-29T16:13:58.260 回答
3

您可以FlagsAttribute在枚举上使用并免费获得许多功能(无需在位级别工作)。

MSDN 将属性描述为:

表示可以将枚举视为位域;即一组标志。

于 2011-03-29T16:13:47.557 回答
2

允许您提取所有[FlagsAttribute]有效值。

http://msdn.microsoft.com/en-us/library/system.flagsattribute.aspx

于 2011-03-29T16:14:12.173 回答
2

尝试这个:

TEnum[] EnumSplit<TEnum>(int mask)
{
    List<TEnum> values = new List<TEnum>();
    foreach(int enumValue in Enum.GetValues(typeof(TEnum)))
    {
        if(mask & enumValue == enumValue)
            values.Add((TEnum)enumValue);
    }
    return values.ToArray();
}

像这样称呼它:

var bar = EnumSplit<foo>(i);

最好将其更改为 returnIEnumerable<TEnum>而不是TEnum[].

于 2011-03-29T16:14:57.697 回答
1

Enum您可以使用从您传递的值中提取值的方法来执行此操作:

public static T[] EnumSplit<T>(int value) where T : struct
{
    // Simplified as Enum.GetValues will complain if T is not an enum
    // However, you should add a check to make sure T implements FlagAttribute
    return (from vv in Enum.GetValues(typeof(T))
            where ((int)vv & value) != 0
            select (T)vv).ToArray();;
}
于 2011-03-29T16:16:16.890 回答
0
    var flags = Enum.GetValues(typeof (/* YourEnumType */))
                  .Cast</* YourEnumType */>()
                  .Select(v => enumValue.HasFlag(v))
                  .ToArray();
于 2011-03-29T16:17:35.003 回答