define 在C语言中的作用,c语言中的define用法
作为代码中,第一个看到的,极有可能就是define这个东西,称为宏!(define是可以出现在任何地方的,但是我们一般把这个写到最开始)然而,很多时候,初学者有时候可能看不懂她,因此,我的c语言学习的第一篇就写这个啦。
define基本用法,简单定义
最浅显的,define能用一个有含义的字符来替代一些数字,比如
#define PI 3.141592654
这样,假如以后要计算圆的周长或者面积,就可以用PI这个字符而不用写3.141592654啦。
比如
#define PI 3.141592654
#include "stdio.h"
int main(){
int r = 3;
float s;
s = PI*r*r;
printf("%f",s);
}
带参数的define
事实上,你可以用define定义很多东西,比如
#define IF(x) if(x){
#define ENDIF }
#include "stdio.h"
int main(){
IF(1)
printf("%d",1);
ENDIF
}
为什么可以这样定义?实际上define的作用仅仅是字符替换而已,所以只要不引起语法错误,没有什么事不可以替换的。为什么会这样?看看下面的
define是怎样工作的
来看看define是怎样工作的,先让我们把上面的有PI的代码保存为test.c,假设你已经安装了gcc,那么执行gcc的预处理命令
gcc -E test.c
你会看到一堆代码,如下
# 1 ""
# 1 "test.c"
......
......
# 3 "test.c" 2
int main(){
int r = 3;
float s;
s = 3.141592654*r*r;
printf("%f",s);
}
看到没,PI在预处理之后就不见了,直接变成了3.141592654
再看看上面的带参数的宏定义的那段代码
# 1 ""
# 1 "test.c"
......
......
# 4 "test.c" 2
int main(){
if(1){
printf("%d",1);
}
}
是不是印证了上面所说的,define实际上只是一个替换的功能而已呢!当然,预处理过程式会做检查的,因此就可以利用这些检查来干一些有意思的事情,这个也是我第一次看这种代码完全看不明白这段代码是什么意思,比如这段
#ifndef COMDEF_H
#define COMDEF_H
......
......
#endif
上面并没有显示的替换啊,这是什么意思!实际上是防止头文件被重复包含啦!仅此而已。
define里面的一些“坑”
define很好用,但是由于仅仅作为替换的她,是有很多坑的。
1.define后面的一些空格
#define SUM (a+b) (a+b)
#define SUM(a+b) (a+b)
define是以第二个空格为分割的,所以第一个其实是错误的。代码中的SUM(1+1) 会被替换为(a+b) (a+b)(1+1)
2.运算符优先级问题
如下代码
#define SUM(x,y) x+y
#include "stdio.h"
int main(){
int a = SUM(2,2)*10;
printf("%d",a);
return 0;
}
我们预期的结果肯定是40了,SUM(2,2)为4,乘上10就是40了,但是结果呢,并不是40,而是22,原因是,经过预处理之后,代码实际上变成了这样
int main(){
int a = 2 +2*10;
printf("%d",a);
return 0;
}
所以,以后define里面含有运算的时候,一定要加括号!一定要加括号!一定要加括号!
扩展阅读
写好C语言,漂亮的宏定义很重要,使用宏定义可以防止出错,提高可移植性,可读性,方便性 等等。下面列举一些成熟软件中常用得宏定义:
1,防止一个头文件被重复包含
#ifndef COMDEF_H
#define COMDEF_H
//头文件内容
#endif
2,得到指定地址上的一个字节或字
#define MEM_B( x ) ( *( (byte *) (x) ) )
#define MEM_W( x ) ( *( (word *) (x) ) )
3,求最大值和最小值
#define MAX( x, y ) ( ((x) > (y)) ? (x) : (y) )
#define MIN( x, y ) ( ((x) < (y)) ? (x) : (y) )
4,得到一个field在结构体(struct)中的偏移量
#define FPOS( type, field ) \
/*lint -e545 */ ( (dword) &(( type *) 0)-> field ) /*lint +e545 */
5,得到一个结构体中field所占用的字节数
#define FSIZ( type, field ) sizeof( ((type *) 0)->field )
6,按照LSB格式把两个字节转化为一个Word
#define FLIPW( ray ) ( (((word) (ray)[0]) * 256) + (ray)[1] )
7,按照LSB格式把一个Word转化为两个字节
#define FLOPW( ray, val ) \
(ray)[0] = ((val) / 256); \
(ray)[1] = ((val) & 0xFF)
8,得到一个变量的地址(word宽度)
#define B_PTR( var ) ( (byte *) (void *) &(var) )
#define W_PTR( var ) ( (word *) (void *) &(var) )
9,得到一个字的高位和低位字节
#define WORD_LO(xxx) ((byte) ((word)(xxx) & 255))
#define WORD_HI(xxx) ((byte) ((word)(xxx) >> 8))
10,返回一个比X大的最接近的8的倍数
#define RND8( x ) ((((x) + 7) / 8 ) * 8 )
11,将一个字母转换为大写
#define UPCASE( c ) ( ((c) >= 'a' && (c) <= 'z') ? ((c) - 0x20) : (c) )
12,判断字符是不是10进值的数字
#define DECCHK( c ) ((c) >= '0' && (c) <= '9')
13,判断字符是不是16进值的数字
#define HEXCHK( c ) ( ((c) >= '0' && (c) <= '9') ||\
((c) >= 'A' && (c) <= 'F') ||\
((c) >= 'a' && (c) <= 'f') )
14,防止溢出的一个方法
#define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))
15,返回数组元素的个数
#define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
16,返回一个无符号数n尾的值MOD_BY_POWER_OF_TWO(X,n)=X%(2^n)
#define MOD_BY_POWER_OF_TWO( val, mod_by ) \
( (dword)(val) & (dword)((mod_by)-1) )
17,对于IO空间映射在存储空间的结构,输入输出处理
#define inp(port) (*((volatile byte *) (port)))
#define inpw(port) (*((volatile word *) (port)))
#define inpdw(port) (*((volatile dword *)(port)))
#define outp(port, val) (*((volatile byte *) (port)) = ((byte) (val)))
#define outpw(port, val) (*((volatile word *) (port)) = ((word) (val)))
#define outpdw(port, val) (*((volatile dword *) (port)) = ((dword) (val)))
18,使用一些宏跟踪调试
A N S I标准说明了五个预定义的宏名。它们是:
_ L I N E _
_ F I L E _
_ D A T E _
_ T I M E _
_ S T D C _
如果编译不是标准的,则可能仅支持以上宏名中的几个,或根本不支持。记住编译程序
也许还提供其它预定义的宏名。
_ L I N E _及_ F I L E _宏指令在有关# l i n e的部分中已讨论,这里讨论其余的宏名。
_ D AT E _宏指令含有形式为月/日/年的串,表示源文件被翻译到代码时的日期。
源代码翻译到目标代码的时间作为串包含在_ T I M E _中。串形式为时:分:秒。
如果实现是标准的,则宏_ S T D C _含有十进制常量1。如果它含有任何其它数,则实现是 非标准的。
可以定义宏,例如:
当定义了_DEBUG,输出数据信息和所在文件所在行
#ifdef _DEBUG
#define DEBUGMSG(msg,date)
printf(msg);
printf(“%d%d%d”,date,_LINE_,_FILE_)
#else
#define DEBUGMSG(msg,date)
#endif
19,宏定义防止使用是错误
1.用小括号包含。
例如:#define ADD(a,b) (a+b)
2.用do{}while(0)语句包含多语句防止错误
例如(错误的):
#define DO(a,b) a+b;\
a++;
应用时:
if(…)
DO(a,b); //产生错误
else
解决方法:
#define DO(a,b) do{a+b;\
a++;}while(0)
赞赏
define 在C语言中的作用,c语言中的define用法相关推荐
- sort在c语言中的作用,c语言中sort的用法详解.docx
c语言中sort的用法详解.docx C语言中SORT的用法详解C语言的学习很多是比较复杂的,那么C语言中SORT的用法的用法你知道吗下面学习啦小编就跟你们详细介绍下C语言中SORT的用法的用法,希望 ...
- sprintf在c语言中的作用,c语言中sprintf的用法
c语言中sprintf的用法的用法你知道吗?下面小编就跟你们详细介绍下c语言中sprintf的用法的用法,希望对你们有用. c语言中sprintf的用法的用法如下: sprintf函数的用法 1.该函 ...
- new在c语言中的作用,C语言中new的用法
匿名用户 1级 2012-01-26 回答 new是C++中用于动态内存分配的运算符,在C语言中一般使用malloc函数e79fa5e98193e78988e69d8331333337386663. ...
- c语言new的作用,C语言中new的用法?
温温酱 new是C++中用于动态内存分配的运算符,在C语言中一般使用malloc函数.new有三种用法:new operator.operator new.placement new1.new ope ...
- %s在c语言中有什么作用,c语言中%s的用法
c语言中%s的用法 C语言是计算机软件领域非常经典的编程语言,unix.linux等众多操作系统均是由C语言编写而成.而在硬件控制.底层驱动等应用领域,C语言更是具有不可替代的作用.下面小编就跟你们详 ...
- c语言中{的作用,C语言中Static和Const关键字的作用
C语言中Static和Const关键字的作用 程序的局部变量存在于(堆栈)中,全局变量存在于(静态区 )中,动态申请数据存在于(堆)中.那么关于C语言中Static和Const关键字的作用,你了解多少 ...
- scanf在c语言中的作用,c语言中scanf的基本用法
前言 scanf()是C语言的格式输入函数,和printf函数一样被声明在stdio.h头文件中,它的基本使用很简单: 1 int a; 2 scanf("%d",&a); ...
- int在c语言中的作用,C语言中int,Uint,uint16等有什么区别以及用处
C语言中int,Uint,uint16等有什么区别以及用处.在C中,既然有了int,为什么还要有uint?特别是uint16,uint32等又有什么用?他们有什么区别?" --------- ...
- break在c语言中的应用,c语言中break的用法
C语言中break语句有以下两种用法: 1.当break语句出现在一个循环内时,循环会立即终止,且程序流将继续执行紧接着循环的下一条语句. 2.它可用于终止switch语句中的一个case. 如果使用 ...
最新文章
- linux+xampp搭建WordPress个人网站过程详解
- hibernate 双向n-n
- JavaScript 数组拼接打印_JavaScript 数组方法
- php dechex 补零,PHP dechex()函数
- 装饰器设计模式的应用
- Codeforces Round #756 (Div. 3)
- SilverLight小游戏
- routeChangeSuccess
- python同步远程文件夹_利用python实现两个文件夹的同步
- [转]linux下IPTABLES配置详解
- Python爬虫 糗百段子
- 思科模拟器路由表怎么看_Cisco路由配置教程 Cisco路由器静态路由与默认路由的配置方法图解...
- 解决谷歌chrome浏览器双击没反应,不能启动(亲测好用)
- 服贸会 | 神州信息:5代ModelB@nk伴随银行业科技创新发展
- WIFI和路由器密码破解的方法
- SiliconLab zigbee host移植到IPQ 807X平台
- 计算机文档怎么字符加宽间距,Word2013设置字符间距,如何设置两个字符之间的距离 -电脑资料...
- Android 底部导航栏 BottomNavigationBar
- 基于matlab的雷达和通信系统,基于MATLAB的多功能通信信号源仿真
- Hadoop 调优之Linux操作系统调优篇