枚举值为什么使用1,2,4,8,16,32等2…

本文详细介绍了在 .NET 中如何使用 Flags 属性来创建复合状态的枚举,并提供了如何解析、设置和移除枚举成员的具体示例。


使用1,2,4,8,16,32等2的幂方,防止重复。 
.NET中的枚举我们一般有两种用法,一是表示唯一的元素序列,例如一周里的各天;还有就是用来表示多种复合的状态。这个时候一般需要为枚举加上[Flags]特性标记为位域,例如: 
[Flags 
enum Styles 
ShowBorder = 1,         //是否显示边框 
ShowCaption = 2,        //是否显示标题 
ShowToolbox = 4         //是否显示工具箱 

  这样我们就可以用"或"运算符组合多个状态,例如
myControl.Style = Styles.ShowBorder | Styles.ShowCaption;  
  这时myControl.Style枚举的值将变成 1+2=3,它的ToString()将变成"Styles.ShowBorder , Styles.ShowCaption" 
  这里我们可以解释为什么第三个值ShowToolbox可以为4,5..而不能为3。也就是说它的值不应该是前几项值的复合值。有一个比较简单的方法就是用2的n次方来依次为每一项赋值,例如 1,2,4,8,16,32,64..... 
  现在举个常见的Flags应用例子。例如一个简单的权限系统,有"Admin"和"User"两种角色,我们可以在表中放一个varchar()字段,以文本形式存放权限字"Admin,User"。但是用Flags型枚举的话,我们就可以直接将 Roles.Admin | Roles.User 的值放在一个int字段里。 
  以下是关于枚举的一些常见操作: 
  将枚举的值变回枚举对象: 
  Styles style = (StylesEnum.Parse(typeof(Styles), 4 );    // -> style = Styles.Toolbox; 
  检查枚举是否包含某个元素: 
  bool hasFlag = ((style & Styles.ShowBorder) != 0); 
  其实我们还会碰到一种情况,就是需要从组合状态中去掉一个元素。用"^"运算符可以做到:
  Styles style = Styles.ShowBorder | Styles.ShowCaption;  
  style = style ^ Styles.ShowBorder; 
  这个时候style的值就会变成 Styles.ShowCaption 
  但这里有一个很严重的问题(偶现在才发现) 
  我们这个时候再执行一次 
  style = style ^ Styles.ShowBorder; 
  按照我们的设想,这个时候 style 的值是 Styles.ShowCaption,不包含 Styles.ShowBorder,所以我们就算去掉这个元素,style应该还是不会变。但实际的 style 的值却又变成了 Styles.ShowBorder | Styles.ShowCaption !! 再执行一遍,又会去掉这个元素,周而复始。 
  当然我们可以在去掉某个元素前做一番检查,如果枚举包含这个元素,再去掉它:
if ((style & Styles.ShowBorder) != 0){  
    style = style ^ Styles.ShowBorder;  

  不知道有没有其它方法可以方便地从Flags枚举状态中去掉一个元素。。 
  Thanks to mobilebilly: 
  style = style & (~Styles.ShowBorder) 可以方便去掉一个元素。 
 (2)枚举成员
        枚举成员是该枚举类型的命名常数。每个枚举成员名称唯一且均具有相关联的常数值,此值的类型就是枚举的基础类型。每个枚举成员的常数值必须在该枚举的基础类型的范围之内。
    示例:
        public enum TimeofDay:uint
        {
            Morning=-3,
            Afternoon=-2,
            Evening=-1
         
    产生编译时错误,原因是常数值 -1、-2 和 –3 不在基础整型 uint 的范围内。
    注意:为所有的每句成员加上注释是很必要的,否则别人看到你的成员里面那么多不认识的单词甚至缩写有多么痛苦……
(3)枚举成员赋值
        规则:允许多个枚举成员有相同的值,允许无顺序地定义成员值,没有显式赋值的成员的值默认为上一个成员的值+1(第一个成员如果没赋值的话默认就是0)。另外,枚举值都不能超过它的基础类型范围。
      示例
        public enum Color
        {
            red = 1,
            green,
            orange = 1,
            grey = 5,
            purple
        }
        上面green的值为2,而purple的值则为6。 red和orange同为1。
    注意:正是由于默认+1这样的规则,建议每个成员都手动赋值,以免出现程序写到一半,从前面改动或者增加成员的时候,后面的成员默认值都会改变,这就麻烦了。
(4)枚举类型与基础类型的转换
     基础类型不能隐式转换为枚举类型
     枚举类型也不能隐式转换为基础类型
    示例:
    public enum Number
    {
        a,
        b
    }
    class Test
    {
        public static void Main()
        {
            int i = Number.a;//错误,要强制类型转换(int)Number.a
            Number n;
            n = 2;//又错误,要强制类型转换(Number)2
        }
    }
(5)枚举类型的使用
switch(timeofDay) //switch中的使用
{
   case TimeofDay.Morning:
        Console.WriteLine("good morning");
        break;
        //……其他的省略啦
}
int hashCode = timeofDay.Morning.GetHashCode();//得到枚举成员的值
string str = timeofDay.Morning.toString();//得到枚举成员的名字
string enumParseStr = Enum.Parse(typeof(timeofDay) , hashCode.ToString()).toString();//将枚举的值转换成对应的名字
int enumParseInt = Convert.ToInt32(Enum.Parse(typeof(timeofTime), str));//将枚举的名字转换成对应的值
<think>我们正在处理一个关于CommonAPI的问题,用户想知道如何将枚举的类型从默认的uint8改为uint16。根据引用[1]:Franca枚举将映射到从基类继承的C++结构CommonAPI::Enumeration。默认情况下,Enum支持的数据类型和连接格式为uint32_t。如果需要,可以通过CommonAPI部署文件*.fdepl文件(枚举支持类型)指定连接格式。注意:引用[1]中提到默认是uint32_t,但用户说默认是uint8。这里可能有一些混淆。实际上,根据CommonAPI的文档,Franca中枚举的默认底层类型是uint32_t。但用户提到uint8,可能是特定版本或配置?不过,引用[1]明确说明了可以通过部署文件来指定枚举的支持类型。因此,我们需要在部署文件(.fdepl)中为枚举指定连接格式(即底层类型)为uint16。步骤:1.在FrancaIDL中定义枚举2.在对应的部署文件(.fdepl)中,使用`Enumeration`元素并设置其`width`属性为16(即uint16)。示例:假设有一个FrancaIDL文件(example.fidl):```interfaceexample{enumMyEnum{VALUE1,VALUE2}methodmyMethod{in{MyEnuminput}}}```然后,在部署文件(example.fdepl)中,我们可以为这个枚举指定底层类型:```xml<?xmlversion="1.0"encoding="UTF-8"?><deploymentxmlns="http://www.generic-robotics.org/franca"><interfacename="example"><enumerationname="MyEnum"><width>16</width></enumeration></interface></deployment>```这样,生成的代码中,枚举的底层类型就会是uint16_t。注意:在部署文件中,`<width>`的有效可以是8、16、32、64,分别对应uint8_t,uint16_t,uint32_t,uint64_t。因此,用户只需要在部署文件中为相应的枚举设置`<width>16</width>`即可。另外,引用[2]和引用[3]似乎与reinterpret_cast和定时器池相关,与当前问题关联不大,因此我们主要参考引用[1]。回答结构:1.解释默认情况(根据引用[1])以及如何通过部署文件更改。2.给出具体步骤和示例。同时,根据系统指令,我们需要生成相关问题。生成的相关问题:1.除了枚举,部署文件还能配置哪些类型?2.如何在CommonAPI中配置结构体(struct)的序列化?3.如何验证部署文件配置已经生效?</think>### 如何在CommonAPI中将枚举类型设置为uint16 根据CommonAPI规范,Franca枚举默认映射为`uint32_t`类型(引用[1])。但您可以通过**部署文件(*.fdepl)** 显式指定枚举的底层数据类型为`uint16_t`,步骤如下: #### 操作步骤 1. **在Franca IDL中定义枚举** 例如在`Example.fidl`中: ```franca enumeration MyEnum { VALUE1, VALUE2, VALUE3 } ``` 2. **在部署文件(.fdepl)中配置宽度** 创建`Example.fdepl`文件,添加`<width>`标签指定16位: ```xml <deployment> <interface name="Example"> <enumeration name="MyEnum"> <width>16</width> <!-- 指定uint16_t --> </enumeration> </interface> </deployment> ``` #### 关键说明 - **`<width>`的有效**: `8`(uint8)、`16`(uint16)、`32`(uint32,默认)、`64`(uint64) - **代码生成效果**: 生成的C++枚举类将继承自`CommonAPI::Enumeration<uint16_t>`: ```cpp class MyEnum : public CommonAPI::Enumeration<uint16_t> { ... }; ``` - **二进制兼容性**: 通信双方必须使用相同的宽度配置,否则会出现数据解析错误(引用[1])。 #### 验证方法 检查生成的C++头文件,确认基类模板参数: ```cpp // 生成代码示例 namespace commonapi { struct MyEnum : public ::commonapi::Enumeration<uint16_t> { enum EnumValue { VALUE1=0, VALUE2=1, VALUE3=2 }; //... }; } // namespace commonapi ``` > **注意**:修改部署文件后需重新生成绑定代码(如使用`commonapi-generator`工具)。 --- ### 相关问题 1. 如何在CommonAPI中为不同枚举分别指定不同的底层整数类型? 2. 部署文件(.fdepl)还能配置哪些通信相关的参数? 3. 如果通信双方配置的枚举宽度不一致(如uint8 vs uint16),会发生什么错误? 4. CommonAPI中是否支持自定义枚举序列化格式(如大端/小端)?
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值