宏定义#define详解

1. # define主要功能

image
  • c语言中用到很多宏定义 ,为了头文件被重复用到#ifndef #define,#ifdefine edfif等条件编译
  • 宏不是语句,结尾不用加“;”,否则会被替换进进程中
  • "#'' 表示这是一条预处理指令
  • 如果写宏不止一行,则在结尾加反斜杠使多行能连接上,但第二行要对齐,否则空格也会作为替换文本的一部分

2.无参宏

  • 无参宏即宏名之后不带参数,只是简单的文本替换
  • typedef与#define区别
    • 两者都可以表示数据类型 1 #define INIT1 int 2 typedef in UINT2
    • 但有时候也会有区别比如定义指针类型的变量时
#define INT1 int *
typedef int * INT2;
INT1 a1, b1;
INT2 a2, b2;
INT1 a1,b1;被替换后为 int *a1,b1,
即一个指向int类型的指针一个int类型的变量;
INT2  a2,b2;   则是两个指向int类型的指针;

3.有参宏

  • 带参数的宏定义,宏名中不能有空格,宏名与形参表之间也不能有空格,而形参表中形参之间可以有空格
    • 有参宏实现一个数的平方
#define COUNT(M) M*M 
int x=6;
print(COUNT(x+1));
print(COUNT(++X));
结果输出:13   和56,而不是函数的输出49
  • 原因在于预编译器不进行技术,只是进行简单的文本替换,COUNT(x+1)被替换成COUNT(x+1*x+1);
  • CUNT(++x)被替换成++x++x即为78=56而不是想要的7*7=49,连续前置自加加两次;
  • 解决办法是用括号将整个替换文本及每个参数用括号括起来但即便是加上括号也不能解决第二种情况,所以解决办法是尽量不使用++,-等符号;
  • 函数实现一个数的平方
int count(int x)
{
return x*x
}

  • 有参宏与函数区别
    • 在宏定义#define COUNT(M) M*M中的形参不分配内存单元,不做类型定义,只是简单的文本替换,而函数int count(int x)中形参x是局部变量,会在栈区分配内存单元,所以要做类型定义,而且实参与形参之间是值传递。而宏只是符号代换,不存在值传递。

  • 宏定义也可以定义表达式或多个语句
#define AB(a,b) a=i+5;b=j+3;   宏定义多个语句
int i=3,j=5;
int m-0,n=0;
AB(m,n);       //宏替换后为m=i+5,n=j+3;
print("m=%d,n=%d",m,n);
输出结果:m=8,n=8

4.#运算符

  • '#' 的作用就是将#后边的宏参数进行字符串的操作,也就是将#后边的参数两边加上一对双引号使其成为字符串。例如a是一个宏的形参,则替换文本中的#a被系统转化为"a",这个转换过程即为字符串化。
#define TEST(param) #param
char *pStr=TEST(123);
printf("pSrt=%s\n",pStr);
输出结果为字符  ”123“

5.##运算符

  • '##' 运算符也可以用在替换文本中,它的作用起到粘合的作用,即将两个宏参数连接为一个数
#define TEST(param1,param2) (param1##param2)
int num =TEST(12,34);
printf("num=%d\n",num);
输出结果为:num=1234

6.可变宏:...和VA_ARGS

  • 作用主要是为了方便管理软件中的打印信息。在写代码或DEBUG时通常需要将一些重要参数打印出来,但在软件发行的时候不希望有这些打印,这时就用到可变参数宏了。
# define PR(...) printf(_VA_ARGS_)
PR("hello\n");
输出结果:hello

*   在宏定义中,形参列表的最后一个参数为省略号...,而“_VA_ARGS_”被用在替换文本中,来表示省略号...代表了什么。


#define PR1(X,...) printf("Message"#X":"_VA_ARGS_)
double mes=10;
PR1(1,"msg=%.2f\n",msg);

输出结果:Message1:msg=10.00

宏替换后为:printf(Message""1""msg=%.2f\n",msg);
  • 需要注意的是:省略号...只能用来替换宏的形参列表中的最后一个。如#define PR1(x,...,y)则是错误的。

7.成熟软件中常用的宏定义。

  • 防止头文件被重复包含
#ifndef COMDEF_H 
#define COMDEF_H
//头文件的内容

#endif
  • 得到一个制定地址上的一个字节或字
 #define MEM_B(X) (*((byte*)(x)))
 #define MEM_W(X) (*((word*)(x)))
  • 求最大值与最小值
 #define MAX(x,y)  ((x)>(y)?(x):(y))
 #define MIN(x,y)  ((x)<(y)?(x):(y))
  • 得到一个结构体中field所占用的字节数
 #define FSIZ(type,field)  sizeof(((type*)0)->field)
  • 得到一个field在结构体中的偏移量
 #define FPOS(type,field)\
                     ((dword)&(((type*)0)->field)
  • 按照LSB格式把两个字节转化为一个word
 #define FLIPW(ray) (((word)(ray)[0]*256)+(ray)[1])
  • 按照LSB格式将一个WORD转化为两个字节
 #define FLOPW(ray,val)  (ray)[0]=((val)/256);(ray)[1]=((val)&0xFF)
  • 得到一个变量的地址
 #define B_PTR(var) ((byte*)(void*)&(var))
 #define W_PTR(var) ((word*)(void*)&(var))
  • 得到一个字的高位与低位字节
 #define WORD_LO(xxx) ((byte)((word)(xxx)&255))
 #define WORD_HI(xxx) ((byte)((word)(xxx)>>8))
  • 用宏得到一个数组所含的元素个数
 #define ARR_SIZE(a) (sizeof(a)/sizeof((a)[0]))
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容