#define是C语言中提供的宏定义命令,其主要目的是为程序员在编程时提供一定的方便,并能在一定程度上提高程序的运行效率,但学生在学习时往往不能 理解该命令的本质,总是在此处产生一些困惑,在编程时误用该命令,使得程序的运行与预期的目的不一致,或者在读别人写的程序时,把运行结果理解错误,这对 C语言的学习很不利。
1 #define命令剖析
1.1 #define的概念
该命令有两种格式:一种是简单的宏定义,另一种是带参数的宏定义。
(1)简单的宏定义:
- #define
<宏名> <字符串>
- 例: #define PI 3.1415926
- #define
<宏名>
(<参数表>)
<宏体>
- 例: #define A(x) x
1.2 宏替换发生的时机
(1)文件包含
可以把源程序中的#include 扩展为文件正文,即把包含的.h文件找到并展开到#include 所在处。
(2)条件编译
预处理器根据#if和#ifdef等编译命令及其后的条件,将源程序中的某部分包含进来或排除在外,通常把排除在外的语句转换成空行。
(3)宏展开
预处理器将源程序文件中出现的对宏的引用展开成相应的宏 定义,即本文所说的#define的功能,由预处理器来完成。
经过预处理器处理的源程序与之前的源程序有所有不同,在这个阶段所进行的工作只是纯粹的替换与展开,没有任何计算功能,所以在学习#define命令时只要能真正理解这一点,这样才不会对此命令引起误解并误用。
2 #define使用中的常见问题解析
- 例1 #define N 2+2
- void main()
- {
- int a=N*N;
- printf(“%d”,a);
- }
(2) 问题解析:
(3)解决办法:
- /*将宏定义写成如下形式*/
- #define N
(2+2)
- /*这样就可替换成(2+2)*(2+2)=16*/
2.2 带参数的宏定义出现的问题
- #define area(x) x*x
- /*这在使用中是很容易出现问题的,看如下的程序*/
- void main()
- {
- int y = area(2+2);
- printf(“%d”,y);
- }
要想能够真正使用好宏定义,那么在读别人的程序时,一定要记住先将程序中对宏的使用全部替换成它所代表的字符串,不要自作主张地添加任何其他符号,完全展开后再进行相应的计算,就不会写错运行结果。
- #include <iostream.h>
- #define product(x) x*x
- int main()
- {
- int i=3;
- int j,k;
- j = product(i++);
- cout<<"j="<<j<<endl;
- cout<<"i="<<i<<endl;
- k = product(++i);
- cout<<"k="<<k<<endl;
- cout<<"i="<<i<<endl;
- return 0;
- }
3 宏定义的优点
(2) 提高程序的运行效率
4 结语
二、define中的三个特殊符号:#,##,#@
- #define Conn(x,y) x##y
- #define ToChar(x)
#@x
- #define ToString(x) #x
- int n
= Conn(123,456);
/* 结果就是n=123456;*/
- char* str = Conn("asdf", "adf"); /*结果就是 str = "asdfadf";*/
做个越界试验char a = ToChar(123);结果就错了;
但是如果你的参数超过四个字符,编译器就给给你报错了!
- #ifndef BODYDEF_H
- #define BODYDEF_H
- //头文件内容
- #endif
- #define MEM_B( x
) (
*( (byte
*)
(x) )
)
- #define MEM_W( x ) ( *( (word *) (x) ) )
- #include
<iostream>
- #include
<windows.h>
- #define MEM_B(x)
(*((byte*)(x)))
- #define MEM_W(x)
(*((WORD*)(x)))
- int main()
- {
- int bTest = 0x123456;
- byte m = MEM_B((&bTest));/*m=0x56*/
- int n = MEM_W((&bTest));/*n=0x3456*/
- return 0;
- }
3 得到一个field在结构体(struct)中的偏移量
- #define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )
4 得到一个结构体中field所占用的字节数
- #define FSIZ( type, field ) sizeof( ((type *) 0)->field )
5 得到一个变量的地址(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) )
- #define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')
- #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]) ) )
ANSI标准说明了五个预定义的宏名。它们是:
- _LINE_ /*(两个下划线),对应%d*/
- _FILE_ /*对应%s*/
- _DATE_ /*对应%s*/
- _TIME_ /*对应%s*/