C语言定义数组时使用枚举作为数组的下标 ——c99功能

部分参考了https://blog.csdn.net/wq3028/article/details/76204690

同时在电脑上进行验证

//温度,电磁阀传感器序号,方便数组定位
typedef enum
{		
	TC314=0,	
	TC355,	
	TC633,		
	TC661,
	TC668,
	TC_NUM_MAX=5,	//共5个温度
	//电磁阀序号,方便数组定位,但要减去TC_NUM_MAX
	SOV311=5,		
	SOV351,				
	SOV461,		
	SOV667,
	SOV455,
	SOV456,
	SOV457,
	SOV458,
	SOV_NUM_MAX=8,	//共8个电磁阀
	
}e_AD8494_TC_NUM;

//保存该电路板中的设备标号,用于对比
const int Device_ID_Table[]=
{
    [TC314]=314, 
	[TC355]=355, 
	[TC633]=633, 
	[TC661]=661, 
	[TC668]=668, //temp
    [SOV311]=311, 
	[SOV351]=351, 
	[SOV461]=461, 
	[SOV667]=667,     //sov
    [SOV455]=455, 
	[SOV456]=456,
	[SOV457]=457, 
	[SOV458]=458
};





const static int Device_ID_Order[] =
{
    TC314, TC355, TC633, TC661, TC668,  //temp
    SOV311, SOV351, SOV461, SOV667,     //sov
    SOV455, SOV456, SOV457, SOV458
};


int size = sizeof(Device_ID_Order)/sizeof(Device_ID_Order[0]);

    for (size_t i = 0; i < size; i++)
    {
        /* code */
        printf("%d\n",Device_ID_Order[i]);
    }

之所以这么写,是因为系统中的变化会随着板子更新而更新,总不能每个工程都在代码中进行一个一个输入,这个操作可以让代码增加扩展性,只要在数组中更改标号即可,而不用在函数中,逐个修改每个语句。

在ANSI的标准确立后,C语言的规范在一段时间内没有大的变动,然而C++在自己的标准化创建过程中继续发展壮大。《标准修正案一》在1994年为C语言创建了一个新标准,但是只修正了一些C89标准中的细节和增加更多更广的国际字符集支持。不过,这个标准引出了1999年ISO 9899:1999的发表。它通常被称为C99。C99被ANSI于2000年3月采用。

在C99中包括的特性有:

  • 增加了对编译器的限制,比如源始码每行要求至少支持到 4095 字节,变量名函数名的要求支持到 63 字节(extern 要求支持到 31)。
  • 增强了预处理功能。例如:
    • 支持取可变参数 #define Macro(...) __VA_ARGS__
    • 使用的时候,允许省略参数,被省略的参数会被扩展成空串。
    • 支持 // 开头的单行注释(这个特性实际上在C89的很多编译器上已经被支持了)
  • 增加了新关键字 restrict, inline, _Complex, _Imaginary, _Bool
    • 支持 long long, long double _Complex, float _Complex 等类型
  • 支持不定长的数组,即数组长度可以在运行时决定,比如利用变量作为数组长度。声明时使用 int a[var] 的形式。不过考虑到效率和实现,不定长数组不能用在全局,或 struct 与 union 。
  • 变量声明不必放在语句块的开头,for 语句提倡写成 for(int i=0;i<100;++i) 的形式,即i 只在 for 语句块内部有效。
  • 允许采用(type_name){xx,xx,xx} 类似于 C++ 的构造函数的形式构造匿名的结构体。
  • 初始化结构的时候允许对特定的元素赋值,形式为:
    struct test{int a[3],b;} foo[] =  { [0].a = {1}, [1].a = 2 };
    
    struct test{int a, b, c, d;} foo =  { .a = 1, .c = 3, 4, .b = 5 };  // 3,4 是对 .c,.d 赋值的
    
  • 格式化字符串中,利用 \u 支持 unicode 的字符。
  • 支持 16 进制的浮点数的描述。
  • printf scanf 的格式化串增加了对 long long int 类型的支持。
  • 浮点数的内部数据描述支持了新标准,可以使用 #pragma 编译器指令指定。
  • 除了已有的 __line__ __file__ 以外,增加了 __func__ 得到当前的函数名。
  • 允许编译器化简非常数的表达式。
  • 修改了 / % 处理负数时的定义,这样可以给出明确的结果,例如在C89中-22 / 7 = -3, -22 % 7 = -1,也可以-22 / 7= -4, -22 % 7 = 6。 而C99中明确为-22 / 7 = -3, -22 % 7 = -1,只有一种结果。
  • 取消了函数返回类型默认为 int 的规定。
  • 允许在 struct 的最后定义的数组不指定其长度,写做 [](flexible array member)。
  • const const int i 将被当作 const int i 处理。
  • 增加和修改了一些标准头文件,比如定义 bool 的 <stdbool.h> ,定义一些标准长度的 int 的 <inttypes.h> ,定义复数的 <complex.h> ,定义宽字符的 <wctype.h> ,类似于泛型的数学函数 <tgmath.h>, 浮点数相关的 <fenv.h>。 在<stdarg.h> 增加了 va_copy 用于复制 ... 的参数。<time.h> 里增加了 struct tmx ,对 struct tm 做了扩展。
  • 输入输出对宽字符以及长整数等做了相应的支持。

但是各个公司对C99的支持所表现出来的兴趣不同。当GCC和其它一些商业编译器支持C99的大部分特性的时候[4]微软Borland却似乎对此不感兴趣。

为了避免这种隐患可以在定义数组时候使用枚举作为数组的下标,这样即使数据输入混乱,但是只要数组定义时候枚举下标定义和数组成员可以对应正确就可以避免这种错误。

    pUART[index++] = (Device_ID_Table[0] >> 8);         //温度1标号高字节
    pUART[index++] = (uint8_t)Device_ID_Table[0];           //温度1标号低字节
    pUART[index++] = g_Device_Info.Temperature[Device_ID_Order[0]] >> 8;         //温度1高字节
    pUART[index++] = g_Device_Info.Temperature[Device_ID_Order[0]];           //温度1低字节

 

posted @ 2019-08-29 16:37  wdliming  阅读(1126)  评论(0编辑  收藏  举报