你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

解决C 语言中枚举难题

[复制链接]
gaosmile 发布时间:2020-3-31 11:44
在实际应用中,有的变量只有几种可能取值。如人的性别只有两种可能取值,星期只有七种可能取值。在 C 语言中对这样取值比较特殊的变量可以定义为枚举类型。所谓枚举是指将变量的值一一列举出来,变量只限于列举出来的值的范围内取值。
定义一个变量是枚举类型,可以先定义一个枚举类型名,然后再说明这个变量是该枚举类型。

例如:

  1. enum weekday{sun,mon,tue,wed,thu,fri,sat};
复制代码



定义了一个枚举类型名 enum weekday,然后定义变量为该枚举类型。例如:

  1. enum weekday day;
复制代码


当然,也可以直接定义枚举类型变量。例如:
  1. <font color="#003000">
  2. enum weekday{sun,mon,tue,wed,thu,fri,sat} day; </font>
复制代码

其中,sum,mon,…,sat 等称为枚举元素或枚举常量,它们是用户定义的标识符。
需要说明的有以下几点:

枚举元素不是变量,而是常数,因此枚举元素又称为枚举常量。因为是常量,所以不能对枚举元素进行赋值。

枚举元素作为常量,它们是有值的,C 语言在编译时按定义的顺序使它们的值为,1,2,…。

在上面的说明中,sun 的值为 0,mon 的值为 1,…sat 的值为 6,如果有赋值语句

  1. day=mon;
复制代码

则 day 变量的值为 1。当然,这个变量值是可以输出的。例如:

  1. <font color="#003000">
  2. printf ("%d",day);</font><font face="Tahoma">
  3. </font>
复制代码
将输出整数 1。

如果在定义枚举类型时指定元素的值,也可以改变枚举元素的值。例如:

  1. enum weekday{sun=7,mon=1,tue,wed,thu,fri,sat}day;
复制代码

这时,sun 为 7,mon 为 1,以后元素顺次加 1,所以 sat 就是 6 了。


枚举值可以用来作判断。例如:

  1. if (day==mon) {…}
  2. if (day>mon) {…}
复制代码

枚举值的比较规则是:按其在说明时的顺序号比较,如果说明时没有人为指定,则第一个枚举元素的值认作 0。例如,mon>sun,sat>fri。
C 语言教程 ?216?


一个整数不能直接赋给一个枚举变量,必须强制进行类型转换才能赋值。例如:

  1. day=(enum weekday)2;
复制代码

这个赋值的意思是,将顺序号为 2 的枚举元素赋给 day,相当于workday=tue;
从键盘输入一个整数,显示与该整数对应的枚举常量的英文名称。

  1. # include
  2. void main( )
  3. {
  4. enum weekday {sun,mon,tue,wed,thu,fri,sat} day;
  5. int k;
  6. printf("input a number(0--6)");
  7. scanf("%d",&k);
  8. day=(enum weekday)k;
  9. switch(day)
  10. {

  11. case sun: printf("sunday/n");break;
  12. case mon: printf("monday/n");break;
  13. case tue: printf("tuesday/n");break;
  14. case wed: printf("wednesday/n");break;
  15. case thu: printf("thursday/n");break;
  16. case fri: printf("friday/n");break;
  17. case sat: printf("satday/n");break;
  18. default: printf("input error/n");break;
  19. }
  20. }
复制代码


程序运行结果为:

  1. input a number(0--6)1
  2. monday
复制代码

在该程序中,枚举常量与枚举变量可以进行比较,但要输出枚举常量对应的英文单词,不能使用以下语句:

  1. printf(" %s",mon);
复制代码

因为枚举常量 mon 为整数值,而非字符串。

在使用枚举变量时,主要关心的不是它的值的大小,而是其表示的状态。



注:以下全部代码的执行环境为VC++ 6.0
在程序中,可能需要为某些整数定义一个别名,我们可以利用预处理指令#define来完成这项工作,您的代码可能是:

  1. #define MON  1
  2. #define TUE   2
  3. #define WED  3
  4. #define THU   4
  5. #define FRI    5
  6. #define SAT   6
  7. #define SUN   7
复制代码



在此,我们定义一种新的数据类型,希望它能完成同样的工作。这种新的数据类型叫枚举型。
▍1. 定义一种新的数据类型 - 枚举型

以下代码定义了这种新的数据类型 - 枚举型

  1. enum DAY
  2. {
  3.       MON=1, TUE, WED, THU, FRI, SAT, SUN
  4. };
复制代码


(1) 枚举型是一个集合,集合中的元素(枚举成员)是一些命名的整型常量,元素之间用逗号,隔开。

(2) DAY是一个标识符,可以看成这个集合的名字,是一个可选项,即是可有可无的项。
(3) 第一个枚举成员的默认值为整型的0,后续枚举成员的值在前一个成员上加1。
(4) 可以人为设定枚举成员的值,从而自定义某个范围内的整数。
(5) 枚举型是预处理指令#define的替代。

(6) 类型定义以分号;结束。
▍2. 使用枚举类型对变量进行声明

新的数据类型定义完成后,它就可以使用了。我们已经见过最基本的数据类型,如:整型int, 单精度浮点型float, 双精度浮点型double, 字符型char, 短整型short等等。用这些基本数据类型声明变量通常是这样:

  1. char     a; //变量a的类型均为字符型char
  2. char     letter;
  3. int        x,
  4.            y,
  5.            z; //变量x,y和z的类型均为整型int
  6. int       number;
  7. double  m, n;
  8. double  result; //变量result的类型为双精度浮点型double
复制代码


既然枚举也是一种数据类型,那么它和基本数据类型一样也可以对变量进行声明。
方法一:枚举类型的定义和变量的声明分开

  1. enum DAY
  2. {
  3.       MON=1, TUE, WED, THU, FRI, SAT, SUN
  4. };

  5. enum DAY yesterday;
  6. enum DAY today;
  7. enum DAY tomorrow; //变量 tomorrow的类型为枚举型enum DAY
  8. enum DAY good_day, bad_day; //变量good_day和bad_day的类型均为枚举型enum DAY
复制代码


方法二:类型定义与变量声明同时进行

  1. enum //跟第一个定义不同的是,此处的标号DAY省略,这是允许的。
  2. {
  3.     saturday,
  4.     sunday = 0,
  5.     monday,
  6.     tuesday,
  7.     wednesday,
  8.     thursday,
  9.     friday
  10. } workday; //变量workday的类型为枚举型enum DAY
  11. enum week { Mon=1, Tue, Wed, Thu, Fri Sat, Sun} days; //变量days的类型为枚举型enum week
  12. enum BOOLEAN { false, true } end_flag, match_flag; //定义枚举类型并声明了两个枚举型变量
复制代码


方法三:用typedef关键字将枚举类型定义成别名,并利用该别名进行变量声明
  1. <font color="#005000"><font face="Tahoma">
  2. typedef enum workday
  3. {
  4.     saturday,
  5.     sunday = 0,
  6.     monday,
  7.     tuesday,
  8.     wednesday,
  9.     thursday,
  10.     friday
  11. } workday; //此处的workday为枚举型enum workday的别名
  12. workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,也即enum workday</font></font><font color="#003000">
  13. </font>
复制代码


enum workday中的workday可以省略:

  1. typedef enum
  2. {
  3.     saturday,
  4.     sunday = 0,
  5.     monday,
  6.     tuesday,
  7.     wednesday,
  8.     thursday,
  9.     friday
  10. } workday; //此处的workday为枚举型enum workday的别名
  11. workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,也即 enum workday
复制代码

也可以用这种方式:

  1. typedef enum workday
  2. {
  3.     saturday,
  4.     sunday = 0,
  5.     monday,
  6.     tuesday,
  7.     wednesday,
  8.     thursday,
  9.     friday
  10. };
  11. workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,也即 enum workday
复制代码


注意:同一个程序中不能定义同名的枚举类型,不同的枚举类型中也不能存在同名的命名常量。错误示例如下所示:
错误声明一:存在同名的枚举类型

  1. typedef enum
  2. {
  3.     wednesday,
  4.     thursday,
  5.     friday
  6. } workday;
  7. typedef enum WEEK
  8. {
  9.     saturday,
  10.     sunday = 0,
  11.     monday,
  12. } workday;
复制代码


错误声明二:存在同名的枚举成员

  1. typedef enum
  2. {
  3.     wednesday,
  4.     thursday,
  5.     friday
  6. } workday_1;
  7. typedef enum WEEK
  8. {
  9.     wednesday,
  10.     sunday = 0,
  11.     monday,
  12. } workday_2;
复制代码

▍3. 使用枚举类型的变量
3.1 对枚举型的变量赋值。
实例将枚举类型的赋值与基本数据类型的赋值进行了对比:

方法一:先声明变量,再对变量赋值

  1. #include<stdio.h>
  2. /* 定义枚举类型 */
  3. enum DAY { MON=1, TUE, WED, THU, FRI, SAT, SUN };
  4. void main()
  5. {
  6.     /* 使用基本数据类型声明变量,然后对变量赋值 */
  7.     int x, y, z;
  8.     x = 10;
  9.     y = 20;
  10.     z = 30;
  11.     /* 使用枚举类型声明变量,再对枚举型变量赋值 */
  12.     enum DAY yesterday, today, tomorrow;
  13.     yesterday = MON;
  14.     today     = TUE;
  15.     tomorrow  = WED;
  16.     printf("%d %d %d /n", yesterday, today, tomorrow);
  17. }
复制代码
方法二:声明变量的同时赋初值

  1. #include <stdio.h>

  2. /* 定义枚举类型 */

  3. enum DAY { MON=1, TUE, WED, THU, FRI, SAT, SUN };

  4. void main()
  5. {

  6.     /* 使用基本数据类型声明变量同时对变量赋初值 */

  7.     int x=10, y=20, z=30;

  8.     /* 使用枚举类型声明变量同时对枚举型变量赋初值 */

  9.     enum DAY yesterday = MON,

  10.                         today = TUE,

  11.                    tomorrow = WED;

  12.     printf("%d %d %d /n", yesterday, today, tomorrow);
  13. }
复制代码


方法三:定义类型的同时声明变量,然后对变量赋值。

  1. #include <stdio.h>
  2. /* 定义枚举类型,同时声明该类型的三个变量,它们都为全局变量 */
  3. enum DAY { MON=1, TUE, WED, THU, FRI, SAT, SUN } yesterday, today, tomorrow;
  4. /* 定义三个具有基本数据类型的变量,它们都为全局变量 */
  5. int x, y, z;
  6. void main()
  7. {
  8.     /* 对基本数据类型的变量赋值 */
  9.     x = 10;  y = 20;  z = 30;
  10.     /* 对枚举型的变量赋值 */
  11.     yesterday = MON;
  12.     today     = TUE;
  13.     tomorrow  = WED;
  14.     printf("%d %d %d /n", x, y, z); //输出:10 20 30
  15.     printf("%d %d %d /n", yesterday, today, tomorrow); //输出:1 2 3
  16. }
复制代码


方法四:类型定义,变量声明,赋初值同时进行。

  1. #include <stdio.h>
  2. /* 定义枚举类型,同时声明该类型的三个变量,并赋初值。它们都为全局变量 */
  3. enum DAY
  4. {
  5.     MON=1,
  6.     TUE,
  7.     WED,
  8.     THU,
  9.     FRI,
  10.     SAT,
  11.     SUN
  12. }
  13. yesterday = MON, today = TUE, tomorrow = WED;
  14. /* 定义三个具有基本数据类型的变量,并赋初值。它们都为全局变量 */
  15. int x = 10, y = 20, z = 30;
  16. void main()
  17. {
  18.     printf("%d %d %d /n", x, y, z); //输出:10 20 30
  19.     printf("%d %d %d /n", yesterday, today, tomorrow); //输出:1 2 3
  20. }
复制代码


3.2 对枚举型的变量赋整数值时,需要进行类型转换。

  1. #include <stdio.h>

  2. enum DAY { MON=1, TUE, WED, THU, FRI, SAT, SUN };

  3. void main()
  4. {

  5.     enum DAY yesterday, today, tomorrow;

  6.     yesterday = TUE;

  7.     today = (enum DAY) (yesterday + 1); //类型转换

  8.     tomorrow = (enum DAY) 30; //类型转换

  9.     //tomorrow = 3; //错误

  10.     printf("%d %d %d /n", yesterday, today, tomorrow); //输出:2 3 30

  11. }
复制代码


3.3 使用枚举型变量

  1. #include<stdio.h>
  2. enum
  3. {
  4.     BELL          = '/a',
  5.     BACKSPACE = '/b',
  6.     HTAB         = '/t',
  7.     RETURN      = '/r',
  8.     NEWLINE    = '/n',
  9.     VTAB         = '/v',
  10.     SPACE       = ' '
  11. };
  12. enum BOOLEAN { FALSE = 0, TRUE } match_flag;
  13. void main()
  14. {
  15.     int index = 0;
  16.     int count_of_letter = 0;
  17.     int count_of_space = 0;
  18.     char str[] = "I'm Ely efod";
  19.     match_flag = FALSE;
  20.     for(; str[index] != '/0'; index++)
  21.         if( SPACE != str[index] )
  22.             count_of_letter++;
  23.         else
  24.         {
  25.             match_flag = (enum BOOLEAN) 1;
  26.             count_of_space++;
  27.         }
  28.     printf("%s %d times %c", match_flag ? "match" : "not match", count_of_space, NEWLINE);
  29.     printf("count of letters: %d %c%c", count_of_letter, NEWLINE, RETURN);
  30. }
复制代码


输出:

  1. match 2 times
  2. count of letters: 10
  3. Press any key to continue
复制代码

▍4. 枚举类型与sizeof运算符

  1. #include <stdio.h>
  2. enum escapes
  3. {
  4.     BELL      = '/a',
  5.     BACKSPACE = '/b',
  6.     HTAB      = '/t',
  7.     RETURN    = '/r',
  8.     NEWLINE   = '/n',
  9.     VTAB      = '/v',
  10.     SPACE     = ' '
  11. };
  12. enum BOOLEAN { FALSE = 0, TRUE } match_flag;
  13. void main()
  14. {
  15.     printf("%d bytes /n", sizeof(enum escapes)); //4 bytes
  16.     printf("%d bytes /n", sizeof(escapes)); //4 bytes
  17.     printf("%d bytes /n", sizeof(enum BOOLEAN)); //4 bytes
  18.     printf("%d bytes /n", sizeof(BOOLEAN)); //4 bytes
  19.     printf("%d bytes /n", sizeof(match_flag)); //4 bytes
  20.     printf("%d bytes /n", sizeof(SPACE)); //4 bytes
  21.     printf("%d bytes /n", sizeof(NEWLINE)); //4 bytes
  22.     printf("%d bytes /n", sizeof(FALSE)); //4 bytes
  23.     printf("%d bytes /n", sizeof(0)); //4 bytes
  24. }
复制代码


▍5. 综合举例

  1. #include<stdio.h>
  2. enum Season
  3. {
  4.     spring, summer=100, fall=96, winter
  5. };
  6. typedef enum
  7. {
  8.     Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday
  9. }
  10. Weekday;
  11. void main()
  12. {
  13.     /* Season */
  14.     printf("%d /n", spring); // 0
  15.     printf("%d, %c /n", summer, summer); // 100, d
  16.     printf("%d /n", fall+winter); // 193
  17.     Season mySeason=winter;
  18.     if(winter==mySeason)
  19.         printf("mySeason is winter /n"); // mySeason is winter
  20.    
  21.     int x=100;
  22.     if(x==summer)
  23.         printf("x is equal to summer/n"); // x is equal to summer
  24.     printf("%d bytes/n", sizeof(spring)); // 4 bytes
  25.     /* Weekday */
  26.     printf("sizeof Weekday is: %d /n", sizeof(Weekday)); //sizeof Weekday is: 4
  27.     Weekday today = Saturday;
  28.     Weekday tomorrow;
  29.     if(today == Monday)
  30.         tomorrow = Tuesday;
  31.     else
  32.         tomorrow = (Weekday) (today + 1); //remember to convert from int to Weekday
  33. }
复制代码
收藏 评论0 发布时间:2020-3-31 11:44

举报

0个回答

所属标签

STM32团队

意法半导体微控制器和微处理器拥有广泛的产品线,包含低成本的8位单片机和基于ARM® Cortex®-M0、M0+、M3、M4、M33、M7及A7内核并具备丰富外设选择的32位微控制器及微处理器


最新内容

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版