C/C++之define用法小结

1、define——(无参数宏定义)用法
 
一般形式为:#define
标识符 字符串
 
(1)“#”表示其为预处理命令,凡是以“#”开头的都是预处理命令;“define”为宏定义命令;“标识符”为所定义的宏名;“字符串”可以是常数、表达式、字符串等。
 
(2)除了经常用到的如“#define MAXNUM 100”之外,还有其它多种灵活的用法,如“#define M (y*y+3*y)”,注意,的表达式中()是必须的,否则在进行如“2*M+2”的运算
时就会出现错误。
 
(3)末尾不需要分号(如果加分号,会连同分号一起代换)。
 
(4)define只是做简单的文本替换。
 
(5)作用域为宏定义开始,到源程序结束,终止定义域可用“#undef M”。
 
(6)宏名如果以字符串的形式被调用,则不做替换,如printf(“I M O”)。
 
(7)可以嵌套进行定义,如
 
#define PI 3.14
 
#define S PI*R*R
 
(8)习惯上把宏名写为大写的,以区别普通的变量。
 
2、define和typedef的区别
 
define宏定义是在预处理完成的,typedef实在编译时处理的,typedef不是简单的代换,而是对类型说明符的重命名。
 
例如:#define P1 int*
 
    typedef int* P2;
 
    P1 a, b;//相当于int* a, b,此时a是int指针,b是int整数。
 
    P2 a, b;//表示a和b都是int指针。
 
3、define(带参数宏定义)用法
 
一般形式为:#define 宏名(形参) 字符串
 
最长见的 #define MAX(a,b) (a>b)?a:b
 
(1)宏名和形参之间不能有空格。如果上式写为 #define MAX (a,b) (a>b)?a:b,则MAX就表示整个后面的部分了。
 
(2)带参宏定义的形参是不分配内存的。
 
        (3) 在宏定义中的形参是标识符,而宏调用中的实参可以是表达式。
 
#define SQ(y) (y)*(y)
 
main(){
 
  int a,sq;
 
  printf("input a number:    ");
 
  scanf("%d",&a);
 
  sq=SQ(a+1);
 
  printf("sq=%d\n",sq);
 
}
 
 
 
上例中第一行为宏定义,形参为y。程序第七行宏调用中实参为a+1,是一个表达式,在宏展开时,用a+1代换y,再用(y)*(y) 代换SQ,得到如下语句 :
 
    sq=(a+1)*(a+1);
 
这与函数的调用是不同的,函数调用时要把实参表达式的值求出来再赋予形参。而宏代换中对实参表达式不作计算直接地照原样代换。
 
(4) 在宏定义中,字符串内的形参通常要用括号括起来以避免出错。在上例中的宏定义中(y)*(y)表达式的y都用括号括起来,因此结果是正确的。如果去掉括号,把程序改 为以下形式:
 
#define SQ(y) y*y
 
main(){
 
  int a,sq;
 
  printf("input a number:    ");
 
scanf("%d",&a);
 
sq=SQ(a+1);
 
  printf("sq=%d\n",sq);
 
}
 
 
 
运行结果为:
 
input a number:3
 
sq=7
 
(5)define的多行定义
define可以替代多行的代码,例如MFC中的宏定义
#define MACRO(arg1, arg2) do { /
/* declarations */ /
stmt1; /
stmt2; /
/* ... */ /
} while(0) /* (no trailing ; ) */
关键是要在每一个换行的时候加上一个"/"
 
4、define宏和函数的区别
 
(1)宏定义可以帮助我们防止出错,提高代码的可移植性和可读性等。
看一个例子,比较两个数或者表达式大小,首先我们把它写成宏定义:
#define MAX( a, b) ( (a) > (b) (a) : (b) )
其次,把它用函数来实现:
int max( int a, int b)
{
return (a > b a : b)
}
很显然,我们不会选择用函数来完成这个任务,原因有两个:
首先,函数调用会带来额外的开销,它需要开辟一片栈空间,记录返回地址,将形参压栈,从函数返回还要释放堆栈。这种开销不仅会降低代码效率,
而且代码量也会大大增加,而使用宏定义则在代码规模和速度方面都比函数更胜一筹;其次,函数的参数必须被声明为一种特定的类型,所以它只能在
类型合适的表达式上使用,我们如果要比较两个浮点型的大小,就不得不再写一个专门针对浮点型的比较函数。反之,上面的那个宏定义可以用于整
形、长整形、单浮点型、双浮点型以及其他任何可以用“>”操作符比较值大小的类型,也就是说,宏是与类型无关的。
和使用函数相比,使用宏的不利之处在于每次使用宏时,一份宏定义代码的拷贝都会插入到程序中。除非宏非常短,否则使用宏会大幅度增加程序的长
度。
还有一些任务根本无法用函数实现,但是用宏定义却很好实现。比如参数类型没法作为参数传递给函数,但是可以把参数类型传递给带参的宏。
看下面的例子:
 
#define MALLOC(n, type) /
 
              
( (type *) malloc((n)* sizeof(type)))
利用这个宏,我们就可以为任何类型分配一段我们指定的空间大小,并返回指向这段空间的指针。我们可以观察一下这个宏确切的工作过程:
int *ptr;
ptr = MALLOC ( 5, int );
将这宏展开以后的结果:
ptr = (int *) malloc ( (5) * sizeof(int) );
这个例子是宏定义的经典应用之一,完成了函数不能完成的功能,但是宏定义也不能滥用,通常,如果相同的代码需要出现在程序的几个地方,更好的
方法是把它实现为一个函数。
 
 
  (2)下面总结和宏和函数的不同之处,以供大家写代码时使用,这段总结摘自《C和指针》一书。
 
代码长度
#define宏:每次使用时,宏代码都被插入到程序中。除了非常小的宏之外,程序的长度将大幅度增长
函数:函数代码只出现于一个地方:每次使用这个函数时,都调用那个地方的同一份代码
 
执行速度
#define宏:更快
函数: 存在函数调用、返回的额外开销
 
操作符优先级
#define宏:宏参数的求值是在所有周围表达式的上下文环境里,除非它们加上括号,否则邻近操作符的优先级可能产生不可预料的结果。
函数:函数参数只在函数调用时求值一次,它的结果值传递给函数。表达式的求值结果更容易预测。
 
 
参数求值
#define宏:参数用于宏定义时,每次都将重新求值,由于多次求值,具有副作用的参数可能会产生不可预测的结果。
函数:参数在函数调用前只求值一次,在函数中多次使用参数并不会导致多次求值过程,参数的副作用并不会造成任何特殊问题。
 
 
参数类型
#define宏:宏与类型无关,只要参数的操作是合法的,它可以用于任何参数类型。
函数: 函数的参数是与类型有关系的,如果参数的类型不同,就需要使用不同的函数,即使它们执行的任务是相同的。

#if、#ifdef、#if defined之间的区别


#if的使用说明


#if的后面接的是表达式


#if (MAX==10)||(MAX==20)
 code...
#endif
它的作用是:如果(MAX==10)||(MAX==20)成立,那么编译器就会把其中的#if 与 #endif之间的代码编译进去(注意:是编译进去,不是执行!!)


#if defined的使用


#if后面接的是一个宏。


#if defined (x)
    ...code...
#endif
这个#if defined它不管里面的“x”的逻辑是“真”还是“假”它只管这个程序的前面的宏定义里面有没有定义“x”这个宏,如果定义了x这个宏,那么,编译器会编译中间的…code…否则不直接忽视中间的…code…代码。


另外 #if defined(x)也可以取反,也就用 #if !defined(x)


#ifdef的使用


#ifdef的使用和#if defined()的用法一致
#ifndef又和#if !defined()的用法一致。


最后强调两点:
第一:这几个宏定义只是决定代码块是否被编译!
第二:别忘了#endif

二、define中的三个特殊符号:#,##,#@

  1. #define Conn(x,y) x##y
  2. #define ToChar(x) #@x
  3. #define ToString(x) #x
(1)x##y表示什么?表示x连接y,举例说:
  1. int n = Conn(123,456); /* 结果就是n=123456;*/
  2. char* str = Conn("asdf", "adf"); /*结果就是 str = "asdfadf";*/
(2)再来看 #@x ,其实就是给x加上单引号,结果返回是一个const char。举例说:
char a = ToChar(1);结果就是a='1';
做个越界试验char a = ToChar(123);结果就错了;
但是如果你的参数超过四个字符,编译器就给给你报错了!
error C2015: too many characters in constant   :P
(3)最后看看#x,估计你也明白了,他是给x加双引号
char* str = ToString(123132);就成了str="123132";
三、常用的一些宏定义

1 防止一个头文件被重复包含 
  1. #ifndef BODYDEF_H 
  2. #define BODYDEF_H 
  3.  //头文件内容 

  4. #endif
 
2 得到指定地址上的一个字节或字

  1. #define MEM_B( x ) ( *( (byte *) (x) ) ) 
  2. #define MEM_W( x ) ( *( (word *) (x) ) )
用法如下:
  1. #include <iostream>
  2. #include <windows.h>

  3. #define MEM_B(x) (*((byte*)(x)))
  4. #define MEM_W(x) (*((WORD*)(x)))

  5. int main()
  6. {
  7.     int bTest = 0x123456;

  8.     byte m = MEM_B((&bTest));/*m=0x56*/
  9.     int n = MEM_W((&bTest));/*n=0x3456*/

  10.     return 0;
  11. }

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

 
  1. #define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )
     请参考文章:详解写宏定义:得到一个field在结构体(struct type)中的偏移量

4 得到一个结构体中field所占用的字节数 
  1. #define FSIZ( type, field ) sizeof( ((type *) 0)->field )

5 得到一个变量的地址(word宽度) 
  1. #define B_PTR( var ) ( (byte *) (void *) &(var) ) 
  2. #define W_PTR( var ) ( (word *) (void *) &(var) )
6 将一个字母转换为大写

  1. #define UPCASE( c ) ( ((c) >= ''a'' && (c) <= ''z'') ? ((c) - 0x20) : (c) )
7 判断字符是不是10进值的数字

  1. #define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')
8 判断字符是不是16进值的数字 
  1. #define HEXCHK( c ) ( ((c) >= ''0'' && (c) <= ''9'') ||((c) >= ''A'' && (c) <= ''F'') ||((c) >= ''a'' && (c) <= ''f'') )
9 防止溢出的一个方法

  1. #define INC_SAT( val ) (val = ((val)+> (val)) ? (val)+: (val))
10 返回数组元素的个数 
  1. #define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
11 使用一些宏跟踪调试

ANSI标准说明了五个预定义的宏名。它们是: 
  1. _LINE_ /*(两个下划线),对应%d*/
  2. _FILE_ /*对应%s*/
  3. _DATE_ /*对应%s*/
  4. _TIME_ /*对应%s*/
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值