我有一个看起来像这样的枚举:
enum foo{
a=0,
b=1,
c=2,
d=4
}
建立标志/位掩码很好,但是否可以做类似的事情
int i = 3;
var bar = Enum.Split(foo,i);
导致类似
bar = foo[]{a, b,c};
谢谢。
尝试以下
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
值(尽管对此无所谓)
允许您提取所有[FlagsAttribute]
有效值。
http://msdn.microsoft.com/en-us/library/system.flagsattribute.aspx
尝试这个:
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[]
.
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();;
}
var flags = Enum.GetValues(typeof (/* YourEnumType */))
.Cast</* YourEnumType */>()
.Select(v => enumValue.HasFlag(v))
.ToArray();