枚举按位运算返回错误的值

时间:2016-11-16 15:20:35

标签: c# enums bitwise-operators

我有颜色选择enum,代表红色,蓝色,绿色和无。

[Flags]
public enum SelectedColor
{
    None, Red, Blue, Green
}

当我创建新枚举并将其设置为RedGreen时,请检查是否设置了Blue,它会返回trueBlue从未在任何地方设置过。

例如:

SelectedColor selectedColor = SelectedColor.Red;
selectedColor |= SelectedColor.Green; //Add Green to Selection

//Check if blue is set
Debug.Log("Blue Selected hasFlag? : " + hasFlag(selectedColor, SelectedColor.Blue));

//Check if blue is set
Debug.Log("Blue Selected isSet? : " + isSet(selectedColor, SelectedColor.Blue));

输出:

  

Blue Selected hasFlag? :错误

     

Blue Selected isSet? :是的

hasFlag和isSet函数:

bool hasFlag(SelectedColor source, SelectedColor value)
{
    int s1 = (int)source;
    return Convert.ToBoolean((s1 & Convert.ToInt32(((int)value) == s1)));
}


bool isSet(SelectedColor source, SelectedColor critValue)
{
    //return ((source & critValue) == critValue);
    return ((source & critValue) != 0);
}

如您所见,我的isSet函数返回了错误的值。我已经尝试了return ((source & critValue) == critValue)return ((source & critValue) != 0);但它们都仍然失败了。这应该根据我对SO和this帖子的研究。

我的hasFlag函数没问题,但为什么isSet函数返回错误的值?

请注意我使用的是.NET 3.5,因此我无法使用.NET 4枚举辅助函数,例如HasFlag

2 个答案:

答案 0 :(得分:9)

如果您没有为枚举指定值,则会将序号分配给它们,如下所示:

[Flags]
public enum SelectedColor // WRONG
{
    None  = 0, // 000
    Red   = 1, // 001
    Blue  = 2, // 010
    Green = 3  // 011 <-- Not the next power of two!
}

然后发生这种情况:

selectedColor = SelectedColor.Red; // 001
selectedColor |= SelectedColor.Green; // (001 | 011 ) == 011, which is still Green

您需要为[Flags]枚举使用2的幂,如下所示:

[Flags]
public enum SelectedColor // CORRECT
{
    None  = 0, // 000
    Red   = 1, // 001
    Blue  = 2, // 010
    Green = 4  // 100
}

然后它正常工作:

selectedColor = SelectedColor.Red; // 001
selectedColor |= SelectedColor.Green; // (001 | 100) == 101, which is Red, Green 

答案 1 :(得分:1)

尽管马修的答案是正确的,但这是一个非常好的答案(那里有#{3;}#{3}}的最新版本和#34;我会保持&#34;通用&#34;并且更简单并且采用另一种方式:与图层蒙版一起使用的相同行为。这里的限制,你的枚举不能超过32项。

private void uncleFoo() {
    enum myNATO { Adam, Bravo, Charlie, Delta } //my enum with NATO codes
    int activeNATO = 0; //int to store selected NATO codes

    //some testing here
    activeNATO |= 1 << (int)myNATO.Adam;
    activeNATO |= 1 << (int)myNATO.Bravo;
    activeNATO |= 1 << (int)myNATO.Charlie;
    Debug.Log(activeNATO);
    Debug.Log("has  Adam? " + (activeNATO & 1 << (int)myNATO.Adam));
    Debug.Log("has Delta? " + (activeNATO & 1 << (int)myNATO.Delta));
}

Debug.Log将打印&#34; 1&#34;为亚当和&#34; 0&#34;对于达美,显然。

希望这会有所帮助;)