1

枚举的Flags 属性允许该枚举类型的变量通过位逻辑采用多个值。这就是为什么建议每个枚举成员都有一个 2 或 0 的幂的数值。

但是,C# 不会阻止您定义一个值为 3 的成员。

这就提出了一个问题 - C# 如何组合具有数值不是 0 或 2 的成员的标志?

例如,一个普通的标志枚举会产生类似的东西(取自上面的 MSDN 链接):

// Define an Enum with FlagsAttribute.
[Flags] 
enum MultiHue : short
{
   None = 0,
   Black = 1,
   Red = 2,
   Green = 4,
   Blue = 8
};

static void Main( )
{
  // Display all combinations of values, and invalid values.
  Console.WriteLine("\nAll possible combinations of values with FlagsAttribute:");

  for( int val = 0; val <= 16; val++ )
     Console.WriteLine( "{0,3} - {1:G}", val, (MultiHue)val);
}

//       All possible combinations of values with FlagsAttribute:
//         0 - None
//         1 - Black
//         2 - Red
//         3 - Black, Red
//         4 - Green
//         5 - Black, Green
//         6 - Red, Green
//         7 - Black, Red, Green
//         8 - Blue
//         9 - Black, Blue
//        10 - Red, Blue
//        11 - Black, Red, Blue
//        12 - Green, Blue
//        13 - Black, Green, Blue
//        14 - Red, Green, Blue
//        15 - Black, Red, Green, Blue
//        16 - 16

但是,如果我要向“MultiHue”枚举添加另一个成员,其数值为“3”,则会产生以下结果:

// Define an Enum with FlagsAttribute.
[Flags] 
enum MultiHue : short
{
   None = 0,
   Black = 1,
   Red = 2,
   Lime = 3,
   Green = 4,
   Blue = 8
};

static void Main( )
{
  // Display all combinations of values, and invalid values.
  Console.WriteLine("\nAll possible combinations of values with FlagsAttribute:");

  for( int val = 0; val <= 16; val++ )
     Console.WriteLine( "{0,3} - {1:G}", val, (MultiHue)val);
}

  0 - None
  1 - Black
  2 - Red
  3 - Lime
  4 - Green
  5 - Black, Green
  6 - Red, Green
  7 - Lime, Green
  8 - Blue
  9 - Black, Blue
 10 - Red, Blue
 11 - Lime, Blue
 12 - Green, Blue
 13 - Black, Green, Blue
 14 - Red, Green, Blue
 15 - Lime, Green, Blue
 16 - 16

请注意如何用不同的成员表示不同的组合,例如:

4 是“绿色”,但也可以是“石灰,黑色”,
7 是“石灰,绿色”,而不是之前的“黑色,红色,绿色”
等。

我知道这是一种不好的做法,但我想知道编译器如何决定将哪些成员组合用于某个位组合?

4

2 回答 2

2

这不是关于禁止或允许组合或按位运算符(允许有或没有Flags属性),因为它表明按位是有意义的,并影响如何ToString()工作。

考虑:

public enum Color
{
    Black = 0,
    Blue = 1,
    Green = 2,
    Red = 4
}

这将使我们能够有意义地讨论 8 种颜色,具体取决于是否设置了给定的蓝色、绿色或红色标志。

我们可以在枚举中定义这些组合:

public enum Color
{
    Black = 0,
    Blue = 1,
    Green = 2,
    Cyan = 3,
    Red = 4,
    Magenta = 5,
    Yellow = 6,
    White = 7
}

请注意,这里的功能是相同的:任何一种方式Color.Green | Color.Red的值都相同6,这当然表示黄色。区别在于我们是否给它贴上方便的标签Yellow

我的枚举和你的枚举之间的主要区别在于,我的组合具有某种普遍意义,因为绿光和红光混合成黄色,依此类推。通常,我们只会发现仅命名 1 位值(1、2、4、8……)很有用,但有时命名特定的常见组合对使用枚举的编码人员很有用。一个真实的例子是DateTimeStyles。在该枚举中,AllowLeadingWhite值为 1,AllowTrailingWhite值为 2,AllowInnerWhite值为 4,AllowWhiteSpaces值为 7,作为指示所有这三个标志都已设置的便捷方式。

于 2019-02-06T11:39:34.393 回答
0

如果你用二进制写它,你可以看到。

1 - black - 00000001
2 - red   - 00000010
3 - lime  - 00000011

所以 3 不是石灰,它应该是 blackRed。3是黑色和红色。您不仅可以设置 2 的幂,以便更轻松地使用组合。相反,与黑色和红色相比,您可以与 blackRed 进行比较。

于 2019-02-06T11:36:30.880 回答