1.简单的宏定义

#define MAXTIME 1000

简单的MATIME定义好了,它代表1000,如果在程序中写:

if(i<MAXTIME){...}

编译器在处理这个代码之前会对MAXTIME进行替换为1000。 宏定义类同于简单的文本替换,而不是变量。

2.define的“函数定义”

define可以像函数那样接受一些参数,如下:

#define max(x,y)(x)>(y)?(x):(y);

这个定义将返回两个数中最大的,而且这个函数没有经过类型检查,好像函数模板,但没有模板安全。存在隐患:

#define Add(a,b) a+b;

如果代数式 cAdd(a,b)d,本意为a+b然后去和c、d相乘,而define只是一个简单替换,式子反而变为了: ca + bd ,忽略了运算规则。

#define pin(int*);
pin a,b;

本来是想定义a,b都是int型指针,而一替换变成了 int*a,b;

3.宏的单行定义

#define A(x) T_##x
#define B (x)#@x
#define C (x)#x

我们假设x=1,则有:

A(1): T_1
B(1): '1'
C(1): "1"

3.define的多行定义

define可以替换多行的代码,例如MFC中的宏定义

#define MACRO(arg1, arg2) do{/
    /*declarations*/
stmt1;/
stmt2;/
}while(0) /*(no trailing;)*/

每行换行时加上换行符“/”

4.条件编译

在大规模的开发中,define最重要的功能就是条件编译。也就是本文的用法。

#ifdef UCLINU
...
...
#endif

编译时,可以通过#define设置编译环境

5如何定义宏,取消宏

//定义宏
#define [MarcoName] [Macro Value]
//取消宏
#undef [MacroName]
//普通宏
#define PI(3.14]
//带参数的宏
#define max(a,b)((a)>(b)?(a),(b))
加括号是个好习惯哦

6.头文件(.h)可以被头文件或C文件包含

重复包含(重复定义) 由于头文件包含可以嵌套,那么C文件就有可能包含多次同一个头文件,就可能出现重复定义的问题的。 通过条件编译开关来避免重复包含(重复定义)

例如 #ifndef headerfileXXX #define headerfileXXX … //文件内容 … #endif

7.define命令的一些高级用法

define中的三个特殊符号“#” “##” “#@”,前面提到过但未解释。

#define Conn(x,y) x##y

x##y 表示什么? 表示x连接y,

int n = Conn(123, 456); //表示n=123456 相信你已经猜到了
char* str = Conn("asdf", "adf"); //表示str = "asdfadf"

#define ToChar(x) #@x

@x,就是给x加上单引号,返回一个const char:

char a = ToChar(1) //a='1'

x, 就是加双引号.

下面看一下宏定义常用用法:

防止头文件被重复包含:

#ifdef BODYDEF_H
#define BODYDEF_H
...
#endif

得到指定地址上的一个字节或字:

#define MEM_B( x ) ( *( (byte *) (x) ) )
#define MEM_W( x ) ( *( (word *) (x) ) )

得到一个field在结构体(struct)中的偏移量

#define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )

得到一个结构体中field所占用的字节数

#define FSIZ( type, field ) sizeof( ((type *) 0)->field )

得到一个变量的地址(word宽度)

#define B_PTR( var ) ( (byte *) (void *) &(var) ) 
#define W_PTR( var ) ( (word *) (void *) &(var) )

将一个字母转换为大写

#define UPCASE( c ) ( ((c) >= ''a'' && (c) <= ''z'') ? ((c) - 0x20) : (c) )

判断字符是不是10进值的数字

#define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')

判断字符是不是16进值的数字

#define HEXCHK( c ) ( ((c) >= ''0'' && (c) <= ''9'') ||((c) >= ''A'' && (c) <= ''F'') ||((c) >= ''a'' && (c) <= ''f'') )

防止溢出的一个方法

#define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))

返回数组元素的个数

#define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )

使用一些宏跟踪调试 在调试时,我们可以设置__DEBUG宏,也可以再Makefile中使用-D编译选项设置,

[cpp] view plain copy print?
#define __DEBUG

使用方法为,

[cpp] view plain copy print?
#ifdef __DEBUG  
printf("%s", ...);  
#endif

另外,ANSI C标准中有几个标准预定义宏,前面几个(func...STDC)常用于printf(sprintf)等语句中:

__func__:在源代码中插入当前所在函数名;
__LINE__:在源代码中插入当前源代码行号;

__FILE__:在源文件中插入当前源文件名;

__DATE__:在源文件中插入当前的编译日期

__TIME__:在源文件中插入当前编译时间;

__STDC__:当要求程序严格遵循ANSI C标准时该标识被赋值为1;

__cplusplus:当编写C++程序时该标识符被定义。

其中__cplusplus常用于头文件中,格式如下:

[cpp] view plain copy print?
#ifndef _ZX_FUNC_H  
#define _ZX_FUNC_H

#ifdef __cplusplus  
extern "C" {  
#endif

/* functions */  
char *strdup (const char *s);

#ifdef __cplusplus  
}  
#endif

#endif

extern"C"表示将其中的代码按照C编译方法编译,目的是实现C++与C语言的调用。

C编译与C++编译的区别是:C会将上面strdup编译成_STRDUP符号,而C++会编译成_STRDUP_CHAR,这也是C++为什么能实现函数重载的原因。extern只能出现在C++文件中,一般如上面的方式置于头文件中。 要在C中调用C++代码,需要在C代码中的函数或变量声明为extern类型,在C++中将函数或变量用extern "C"修饰。

简单数学计算(绝对值,三角函数等)

[cpp] view plain copy print?
#define ABS( a ) ( ((a)>0) ? (a) : (-(a)) )

define 一个复杂语句

比如交换a,b的值,

[cpp] view plain copy print?
#define(a,b) do { \  
    int t = 0;  
    t = a; \  
    a = b; \  
    b = t; \  
} while(0)

define的这些高级用法在Linux内核源代码很多处出现,可阅读参考内核源代码。

by 李鹏