簡體   English   中英

枚舉標志屬性C#

[英]Enum flag attribute C#

我看了一些相同的主題,但沒有找到我正在尋找的東西我應該使用flag enum flag atribute並檢查我的數據是否在此枚舉的一個集合中例如,enum:

[Flags]
private enum MyEnum {
Apple,
Orange,
Tomato,
Potato
Melon,
Watermelon,

Fruit = Apple | Orange,
Vegetable = Tomato | Potato,
Berry = Melon | Watermelon,
}

在方法中,我應該檢查輸入數據。 我該怎么做?

private void Checking(string data){
if(MyEnum.Fruit contains data) MessageBox.Show("Fruit");
if(MyEnum.Vegetable contains data) MessageBox.Show("Vegetables");
if(MyEnum.Berry contains data) MessageBox.Show("Berry");
}

應該是什么而不是“包含數據”?

UPDATE

private void ZZZ(){
Cheching("Apple");
}

首先,您需要使用powers-of-2序列手動編號您的值:

[Flags]
private enum MyEnum 
{
  None = 0,   // often useful
  Apple = 1,
  Orange = 2,
  Tomato = 4,
  Potato = 8,
  Melon =  16,
  Watermelon = 32,

  Fruit = Apple | Orange,
  Vegetable = Tomato | Potato,
  Berry = Melon | Watermelon,
}

[Flags]屬性不是嚴格必需的,它只控制ToString()行為。

要檢查字符串是否與您的值匹配,您必須先將其設為枚舉:

private void Checking(string data)
{      
    MyEnum v = (MyEnum) Enum.Parse(typeof(MyEnum), data);

    if((MyEnum.Fruit & v) != 0) MessageBox.Show("It's a Fruit"); 
    ...
}

但是請注意,Enum和字符串之間的交換與Parse()是有限的。

除了Henk Holterman的解決方案,您還可以使用擴展方法

[Flags]
private enum MyEnum {
  None = 0, 
  Apple = 1,
  Orange = 2,
  Tomato = 4,
  Potato = 8,
  Melon = 16,
  Watermelon = 32,

  Berry = Melon | Watermelon, 
  Fruit = Apple | Orange,
  Vegetable = Potato | Tomato
}

private static class MyEnumExtensions { 
  public static Boolean IsFruit(this MyEnum value) {
    return (value & MyEnum.Fruit) == MyEnum.Fruit;
  }

  public static Boolean IsVegetable(this MyEnum value) {
    return (value & MyEnum.Vegetable) == MyEnum.Vegetable;
  }

  public static Boolean IsBerry(this MyEnum value) {
    return (value & MyEnum.Berry) == MyEnum.Berry;
  }
}

...

MyEnum data = ...

if (data.IsBerry()) {
  MessageBox.Show("Berry");       
} 

您還可以使用Enum -class的HasFlag -method。 正如Henk指出的那樣,需要使用powers-of-2序列的值手動為您的枚舉賦值。

[Flags]
private enum MyEnum 
{
    Apple = 1,
    Orange = 2,
    Tomato = 4,
    Potato = 8,
    Melon  16,
    Watermelon = 32,

    Fruit = Apple | Orange,
    Vegetable = Tomato | Potato,
    Berry = Melon | Watermelon,
}

然后,要檢查您是否可以使用以下方法,該方法適用於枚舉的所有組成部分:

void Cheking(string data)
{
    // Get the enum value of the string passed to the method
    MyEnum myEnumData;
    if (Enum.TryParse<MyEnum>(data, out myEnumData))
    {
        // If the string was a valid enum value iterate over all the value of
        // the underlying enum type
        var values = Enum.GetValues(typeof(MyEnum)).OfType<MyEnum>();
        foreach (var value in values)
        {
            // If the value is not a power of 2 it is a composed one. If it furthermore
            // has the flag passed to the method this is one we searched.
            var isPowerOfTwo = (value != 0) && ((value & (value - 1)) == 0);
            if (!isPowerOfTwo && value.HasFlag(myEnumData))
            {
                MessageBox.Show(value.ToString());
            }
        }
    }
    // In case an invalid value had been passed to the method
    // display an error message.
    else
    {
        MessageBox.Show("Invalid Value");
    }
}

或者使用LINQ以更短的方式編寫它:

var results = Enum.GetValues(typeof(MyEnum))
                  .OfType<MyEnum>()
                  .Select(x => new { Value = x, IsPowerOfTwo = (x != 0) && ((x & (x - 1)) == 0) } )
                  .Where(x => !x.IsPowerOfTwo && x.Value.HasFlag(myEnumData))
                  .Select(x => x.Value.ToString());

這將給出包含結果的IEnumerable<string> 如果myEnumData的值為MyEnum.Apple則結果將僅包含值"Fruit"

正如@Henk Holterman建議的那樣,首先需要為枚舉值賦值。
所有值都應該是2的冪(並避免使用0,除了“無”的特殊情況)它應該看起來像這樣:

 MyEnum eVal= (MyEnum ) Enum.Parse( typeof(MyEnum), data, true );
 if((MyEnum.Fruit & eVal) != 0) MessageBox.Show("Fruit");

您可能想要閱讀有關按位和布爾代數的更多信息。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM