#define的用法

始终记住#define的核心语义为替换

一、宏的定义与撤销
#普通宏定义
#define PI 3.14    //编译阶段替换掉宏
#define T1 3+4     //容易产生歧义
#define T2 (3+4)   //添加括号后,语义清楚
 
float r = 1.0;
float area = PI * r * r;    
int a = 2* T1    #宏替换后变成   int a = 2*3+4     不符合本意    
ing a = 2* T2    #红替换后变成   int a = 2*(3+4)   符合本意   
 
#undef PI
float area = PI * r * r;     #error: ‘PI’ was not declared in this scope
 
//引号中的宏定义不会被替换
printf("%s:%f\n", "PI", PI);    //输出 PI:3.14
 
//宏定义的名字必须是合法标识符
#define 0x abcd    //error 不能以数字开始
 
//宏定义中双引号和单引号必须成对出现
#define TEST11 "Z    //error
#define TEST2 'Z     //error
二、带有参数的宏定义
//max和min的宏定义带参数
#define MAX(a,b) (a>b ? a:b)
#define MIN(a,b) (a<b ? a:b)
 
//使用带参数的宏定义
int sum= MAX(1,2) + MIN(1,2);    //替换后语句为:int sum = (1>2 ? 1:2) + (1<2 ? 1:2)
 
//参数个数必须宏定义时形参的个数相同
MAX(1,2,3);    //会报错
 
#undef MAX    //撤销MAX的宏定义
MAX(1,2);    //error: ‘MAX’ was not declared in this scope
三、跨行的宏定义 使用反引号\连接
#定义一个交换数值的多行宏,使用反斜杠连接不同行
#define SWAP(a,b) do { \
    int t = 0;\
    t = a; \
    a = b; \
    b = t; \
} while(0)
四、三个特殊符号:#,##,#@
#define CONNECT(a,b) a##b
#define TOCHAR(a) #@a
#define TOSTRING(a) #a
 
//a##b表示连接
int n = CONNECT(123, 456);                //结果  n = 123456
char *str = CONNECT("abcd", "efg");       //结果  str = "abcdefg"
 
//@#a 表示用单引号包括参数a,返回的是一个字符
char * ch1 = TOCHAR(1);        //结果  ch = '1'
char * ch2 = TOCHAR(123);      //报错,单引号只用在单个字符里
 
//#a 表示用双引号包括参数a,返回一个字符串
char * str1 = TOSTRING(123);    // str = "123"
五、常见的宏定义
  • 防止头文件被重复包含
#ifndef BODYDEF_H 
#define BODYDEF_H 
 
//头文件内容 
 
#endif
  • 得到指定地址上的一个字节值或字值
#include "stdio.h"
//B表示字节byte
#define MEM_B( x )  ( *( (byte *) (x) ) )
//B表示字word,可以理解为int
#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;
}
  • 得到一个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]) ) )
  • 跳转到绝对地址
#define jump(TargetAddr) (*(void(*)())(TargetAddr))()
`#define` 是 C 语言中的一个预处理指令,用于定义宏。宏是一种文本替换机制,可以将代码中的某个标识符替换为指定的文本,从而简化代码的编写和维护。 `#define` 的一般形式为: ```c #define macro_name macro_value ``` 其中,`macro_name` 是宏的名称,`macro_value` 是宏的取值。宏名称通常使用大写字母表示,以便与变量名区分。 例如,可以使用如下的 `#define` 定义一个宏: ```c #define PI 3.1415926 ``` 这个宏的名称是 `PI`,取值是 `3.1415926`。在代码中使用该宏时,编译器会将所有的 `PI` 替换为 `3.1415926`,例如: ```c double r = 2.0; double area = PI * r * r; ``` 经过替换后,上面的代码等价于: ```c double r = 2.0; double area = 3.1415926 * r * r; ``` 除了简单的常量定义,`#define` 还可以定义带参数的宏,例如: ```c #define MAX(a, b) ((a) > (b) ? (a) : (b)) ``` 这个宏的名称是 `MAX`,带有两个参数 `a` 和 `b`,取值是 `(a) > (b) ? (a) : (b)`。在代码中使用该宏时,编译器会将所有的 `MAX(a, b)` 替换为 `(a) > (b) ? (a) : (b)`,例如: ```c int x = 1, y = 2; int z = MAX(x++, y++); ``` 经过替换后,上面的代码等价于: ```c int x = 1, y = 2; int z = (x++ > y++) ? x++ : y++; ``` 需要注意的是,宏的替换是在预处理阶段完成的,替换后的代码会直接参与编译,因此宏定义中的错误会直接导致编译错误,而不是运行时错误。此外,宏的使用也可能会导致一些意料之外的问题,如宏参数的副作用和宏嵌套等,因此需要谨慎使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值