今天在分析一份源码的时候遇到了如下代码:

  1. #define CASE(x,y) case x: _bsd_setlasterror(anApplInstance, BSD_ ## y); break

觉得其中的##很有意思,于是就将这个知识点整理一下。#和##都是预处理指令,我们先看#。
#用来把参数转换成字符串,请看下面的两个例子。
例子一:

  1. 1. #include <stdio.h>
  2. 2. #include <iostream>
  3. 3.
  4. 4. #define P(A) printf("%s:%d\n",#A,A);
  5. 5.
  6. 6. int main(int argc, char **argv)
  7. 7. {
  8. 8. int a = 1, b = 2;
  9. 9. P(a);
  10. 10. P(b);
  11. 11. P(a+b);
  12. 12. system("pause");
  13. 13. }

C语言中的#,##和 __VA_ARGS__            
       typedef enum 和 enum - 图1
例子二:

  1. #define SQR(x) printf("The square of x is %d.\n", ((x)*(x)));

如果这样使用宏:SQR(8);
则输出为:The square of x is 64.
注意到没有,引号中的字符x被当作普通文本来处理,而不是被当作一个可以被替换的语言符号。
假如你确实希望在字符串中包含宏参数,那我们就可以使用“#”,它可以把语言符号转化为字符串。上面的例子改一改:

  1. #define SQR(x) printf("The square of "#x" is %d.\n", ((x)*(x)));

再使用:SQR(8);
则输出的是:The square of 8 is 64.
和#运算符一样,##运算符可以用于宏函数的替换部分。这个运算符把两个语言符号组合成单个语言符号。看例子:

  1. #define XNAME(n) x ## n

如果这样使用宏:XNAME(8)
则会被展开成这样:x8
##就是个粘合剂,将前后两部分粘合起来,也就是有“字符化”的意思。但是“##”不能随意粘合任意字符,必须是合法的C语言标示符。在单一的宏定义中,最多可以出现一次“#”或“##”预处理操作符。如果没有指定与 “#”或“##”预处理操作符相关的计算次序,则会产生问题。为避免该问题,在单一的宏定义中只能使用其中一种操作符(即,一份“#”或一个“##”,或都不用)。除非非常有必要,否则尽量不要使用“#”和“##”。

三、VA_ARGS 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。
四、##VA_ARGS 宏前面加上##的作用在于,当可变参数的个数为0时,这里的##起到把前面多余的”,”去掉的作用,否则会编译出错

C++中typedef enum 和 enum

enum与typedef enum的用法

(2016-05-26 20:35:54)
C语言中的#,##和 __VA_ARGS__            
       typedef enum 和 enum - 图2

在程序中,可能需要为某些整数定义一个别名,我们可以利用预处理指令#define来完成这项工作,您的代码可能是:
#define MON 1
#define TUE 2
#define WED 3
#define THU 4
#define FRI 5
#define SAT 6
#define SUN 7

在此,我们定义一种新的数据类型,希望它能完成同样的工作。这种新的数据类型叫枚举型。
1. 定义一种新的数据类型 - 枚举型
以下代码定义了这种新的数据类型 - 枚举型
enum DAY
{
MON=1, TUE, WED, THU, FRI, SAT, SUN
};
(1) 枚举型是一个集合,集合中的元素(枚举成员)是一些命名的整型常量,元素之间用逗号,隔开。
(2) DAY是一个标识符,可以看成这个集合的名字,是一个可选项,即是可有可无的项。
(3) 第一个枚举成员的默认值为整型的0,后续枚举成员的值在前一个成员上加1。
(4) 可以人为设定枚举成员的值,从而自定义某个范围内的整数。
(5) 枚举型是预处理指令#define的替代。
(6) 类型定义以分号;结束。

2. 使用枚举类型对变量进行声明
新的数据类型定义完成后,它就可以使用了。我们已经见过最基本的数据类型,如:整型int, 单精度浮点型float, 双精度浮点型double, 字符型char, 短整型short等等。用这些基本数据类型声明变量通常是这样:
char a; //变量a的类型均为字符型char
char letter;
int x,
y,
z; //变量x,y和z的类型均为整型int
int number;
double m, n;
double result; //变量result的类型为双精度浮点型double

既然枚举也是一种数据类型,那么它和基本数据类型一样也可以对变量进行声明。
方法一:枚举类型的定义和变量的声明分开
enum DAY
{
MON=1, TUE, WED, THU, FRI, SAT, SUN
};

enum DAY yesterday;
enum DAY today;
enum DAY tomorrow; //变量tomorrow的类型为枚举型enum DAY
enum DAY good_day, bad_day; //变量good_day和bad_day的类型均为枚举型enum DAY

方法二:类型定义与变量声明同时进行:
enum //跟第一个定义不同的是,此处的标号DAY省略,这是允许的。
{
saturday,
sunday = 0,
monday,
tuesday,
wednesday,
thursday,
friday
} workday; //变量workday的类型为枚举型enum DAY

enum week { Mon=1, Tue, Wed, Thu, Fri Sat, Sun} days; //变量days的类型为枚举型enum week

enum BOOLEAN { false, true } end_flag, match_flag; //定义枚举类型并声明了两个枚举型变量

方法三:用**typedef**关键字将枚举类型定义成别名,并利用该别名进行变量声明:
typedef enum workday
{
saturday,
sunday = 0,
monday,
tuesday,
wednesday,
thursday,
friday
} workday; //**
此处的workday为枚举型enum workday的别名

workday today, tomorrow; //变量todaytomorrow的类型为枚举型workday,即enum workday**

也可以:
typedef enum
{
saturday,
sunday = 0,
monday,
tuesday,
wednesday,
thursday,
friday
} workday; //此处的workday为枚举型enum workday的别名

workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,即enum workday

也可以用这种方式:
typedef enum workday
{
saturday,
sunday = 0,
monday,
tuesday,
wednesday,
thursday,
friday
};

workday today, tomorrow; //变量today和tomorrow的类型为枚举型workday,即enum workday

=====================================================================
I haven’t written any C++ in years and now I’m trying to get back into it. I then ran across this and thought about giving up:

  1. typedef enum TokenType
  2. {
  3. blah1 = 0x00000000,
  4. blah2 = 0X01000000,
  5. blah3 = 0X02000000
  6. } TokenType;

What is this? Why is the typedef keyword used here? Why does the name TokenType appear twice in this declaration? How are the semantics different from this:

  1. enum TokenType
  2. {
  3. blah1 = 0x00000000,
  4. blah2=0x01000000,
  5. blah3=0x02000000
  6. };

第一种写法使用比较简单,如TokenType mytype;
如果是第二种写法,则定义变量的时候必须写成:
enum TokenType my_type;
=================================================================

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

typedef enum WEEK
{
saturday,
sunday = 0,
monday,
} workday;

错误声明二:存在同名的枚举成员
typedef enum
{
wednesday,
thursday,
friday
} workday_1;

typedef enum WEEK
{
wednesday,
sunday = 0,
monday,
} workday_2;


3. 使用枚举类型的变量
3.1 对枚举型的变量赋值。
实例将枚举类型的赋值与基本数据类型的赋值进行了对比:
方法一:先声明变量,再对变量赋值
#include<stdio.h>

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

void main()
{

int x, y, z;

x = 10;
y = 20;
z = 30;


enum DAY yesterday, today, tomorrow;

yesterday = MON;
today = TUE;
tomorrow = WED;

  1. printf("%d %d %d \n", yesterday, today, tomorrow);<br />}<br /> <br />方法二:声明变量的同时赋初值<br />#include <</SPAN>stdio.h>

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

void main()
{

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

<br />    enum DAY yesterday = MON, <br />                        today = TUE,<br />                   tomorrow = WED;

printf("%d %d %d \n", yesterday, today, tomorrow);<br />}<br /> <br />方法三:定义类型的同时声明变量,然后对变量赋值。<br />#include <</SPAN>stdio.h>

enum DAY { MON=1, TUE, WED, THU, FRI, SAT, SUN } yesterday, today, tomorrow;

int x, y, z;

void main()
{

x = 10; y = 20; z = 30;


yesterday = MON;
today = TUE;
tomorrow = WED;

printf("%d %d %d \n", x, y, z); //输出:10 20 30<br />    printf("%d %d %d \n", yesterday, today, tomorrow); //输出:1 2 3<br />}<br /> <br />方法四:类型定义,变量声明,赋初值同时进行。<br />#include <</SPAN>stdio.h>

enum DAY
{
MON=1,
TUE,
WED,
THU,
FRI,
SAT,
SUN
}
yesterday = MON, today = TUE, tomorrow = WED;

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

void main()
{
printf(“%d %d %d \n”, x, y, z); //输出:10 20 30
printf(“%d %d %d \n”, yesterday, today, tomorrow); //输出:1 2 3
}

3.2 对枚举型的变量赋整数值时,需要进行类型转换。
#include <stdio.h>

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

void main()
{
enum DAY yesterday, today, tomorrow;

yesterday = TUE;<br />    today = (enum DAY) (yesterday + 1); //类型转换<br />    tomorrow = (enum DAY) 30; //类型转换<br />    //tomorrow = 3; //错误

printf("%d %d %d \n", yesterday, today, tomorrow); //输出:2 3 30<br />}<br /> <br />3.3 使用枚举型变量<br />#include<</SPAN>stdio.h>

enum
{
BELL = ‘\a’,
BACKSPACE = ‘\b’,
HTAB = ‘\t’,
RETURN = ‘\r’,
NEWLINE = ‘\n’,
VTAB = ‘\v’,
SPACE = ‘ ‘
};

enum BOOLEAN { FALSE = 0, TRUE } match_flag;

void main()
{
int index = 0;
int count_of_letter = 0;
int count_of_space = 0;

char str[] = "I'm Ely efod";

match_flag = FALSE;

for(; str[index] != '\0'; index++)<br />        if( SPACE != str[index] )<br />            count_of_letter++;<br />        else<br />        {<br />            match_flag = (enum BOOLEAN) 1;<br />            count_of_space++;<br />        }<br />    <br />    printf("%s %d times %c", match_flag ? "match" : "not match", count_of_space, NEWLINE);<br />    printf("count of letters: %d %c%c", count_of_letter, NEWLINE, RETURN);<br />}<br /> <br />输出:<br />match 2 times<br />count of letters: 10<br />Press any key to continue<br /> <br />4. 枚举类型与sizeof运算符<br />#include<</SPAN>stdio.h>

enum escapes
{
BELL = ‘\a’,
BACKSPACE = ‘\b’,
HTAB = ‘\t’,
RETURN = ‘\r’,
NEWLINE = ‘\n’,
VTAB = ‘\v’,
SPACE = ‘ ‘
};

enum BOOLEAN { FALSE = 0, TRUE } match_flag;

void main()
{
printf(“%d bytes \n”, sizeof(enum escapes)); //4 bytes
printf(“%d bytes \n”, sizeof(escapes)); //4 bytes

printf("%d bytes \n", sizeof(enum BOOLEAN)); //4 bytes<br />    printf("%d bytes \n", sizeof(BOOLEAN)); //4 bytes<br />    printf("%d bytes \n", sizeof(match_flag)); //4 bytes

printf("%d bytes \n", sizeof(SPACE)); //4 bytes<br />    printf("%d bytes \n", sizeof(NEWLINE)); //4 bytes<br />    printf("%d bytes \n", sizeof(FALSE)); //4 bytes<br />    printf("%d bytes \n", sizeof(0)); //4 bytes<br />}<br />5. 综合举例<br />#include<</SPAN>stdio.h>

enum Season //注意这里如果放在下面,后面就不能直接那样声明
{
spring, summer=100, fall=96, winter
};

typedef enum
{
Monday, Tuesday, Wednesday, Thursday, Friday, Saturday, Sunday
}
Weekday;

void main()
{

printf(“%d \n”, spring); // 0
printf(“%d, %c \n”, summer, summer); // 100, d
printf(“%d \n”, fall+winter); // 193

Season mySeason=winter;                //就是这里,看上面的定义,也可以 enum Season mySeason<br />    if(winter==mySeason)<br />        printf("mySeason is winter \n"); // mySeason is winter<br />    <br />    int x=100;<br />    if(x==summer)<br />        printf("x is equal to summer\n"); // x is equal to summer

printf("%d bytes\n", sizeof(spring)); // 4 bytes

<br />    printf("sizeof Weekday is: %d \n", sizeof(Weekday)); //sizeof Weekday is: 4

Weekday today = Saturday;<br />    Weekday tomorrow;<br />    if(today == Monday)<br />        tomorrow = Tuesday;<br />    else<br />        tomorrow = (Weekday) (today + 1); //remember to convert from int to Weekday<br />}<br />分享: