作为代码中,第一个看到的,极有可能就是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用法相关推荐

  1. sort在c语言中的作用,c语言中sort的用法详解.docx

    c语言中sort的用法详解.docx C语言中SORT的用法详解C语言的学习很多是比较复杂的,那么C语言中SORT的用法的用法你知道吗下面学习啦小编就跟你们详细介绍下C语言中SORT的用法的用法,希望 ...

  2. sprintf在c语言中的作用,c语言中sprintf的用法

    c语言中sprintf的用法的用法你知道吗?下面小编就跟你们详细介绍下c语言中sprintf的用法的用法,希望对你们有用. c语言中sprintf的用法的用法如下: sprintf函数的用法 1.该函 ...

  3. new在c语言中的作用,C语言中new的用法

    匿名用户 1级 2012-01-26 回答 new是C++中用于动态内存分配的运算符,在C语言中一般使用malloc函数e79fa5e98193e78988e69d8331333337386663. ...

  4. c语言new的作用,C语言中new的用法?

    温温酱 new是C++中用于动态内存分配的运算符,在C语言中一般使用malloc函数.new有三种用法:new operator.operator new.placement new1.new ope ...

  5. %s在c语言中有什么作用,c语言中%s的用法

    c语言中%s的用法 C语言是计算机软件领域非常经典的编程语言,unix.linux等众多操作系统均是由C语言编写而成.而在硬件控制.底层驱动等应用领域,C语言更是具有不可替代的作用.下面小编就跟你们详 ...

  6. c语言中{的作用,C语言中Static和Const关键字的作用

    C语言中Static和Const关键字的作用 程序的局部变量存在于(堆栈)中,全局变量存在于(静态区 )中,动态申请数据存在于(堆)中.那么关于C语言中Static和Const关键字的作用,你了解多少 ...

  7. scanf在c语言中的作用,c语言中scanf的基本用法

    前言 scanf()是C语言的格式输入函数,和printf函数一样被声明在stdio.h头文件中,它的基本使用很简单: 1 int a; 2 scanf("%d",&a); ...

  8. int在c语言中的作用,C语言中int,Uint,uint16等有什么区别以及用处

    C语言中int,Uint,uint16等有什么区别以及用处.在C中,既然有了int,为什么还要有uint?特别是uint16,uint32等又有什么用?他们有什么区别?" --------- ...

  9. break在c语言中的应用,c语言中break的用法

    C语言中break语句有以下两种用法: 1.当break语句出现在一个循环内时,循环会立即终止,且程序流将继续执行紧接着循环的下一条语句. 2.它可用于终止switch语句中的一个case. 如果使用 ...

最新文章

  1. linux+xampp搭建WordPress个人网站过程详解
  2. hibernate 双向n-n
  3. JavaScript 数组拼接打印_JavaScript 数组方法
  4. php dechex 补零,PHP dechex()函数
  5. 装饰器设计模式的应用
  6. Codeforces Round #756 (Div. 3)
  7. SilverLight小游戏
  8. routeChangeSuccess
  9. python同步远程文件夹_利用python实现两个文件夹的同步
  10. [转]linux下IPTABLES配置详解
  11. Python爬虫 糗百段子
  12. 思科模拟器路由表怎么看_Cisco路由配置教程 Cisco路由器静态路由与默认路由的配置方法图解...
  13. 解决谷歌chrome浏览器双击没反应,不能启动(亲测好用)
  14. 服贸会 | 神州信息:5代ModelB@nk伴随银行业科技创新发展
  15. WIFI和路由器密码破解的方法
  16. SiliconLab zigbee host移植到IPQ 807X平台
  17. 计算机文档怎么字符加宽间距,Word2013设置字符间距,如何设置两个字符之间的距离 -电脑资料...
  18. Android 底部导航栏 BottomNavigationBar
  19. 基于matlab的雷达和通信系统,基于MATLAB的多功能通信信号源仿真
  20. Hadoop 调优之Linux操作系统调优篇

热门文章

  1. d2admin 登陆 笔记
  2. prctl函数的应用
  3. 为什么C++中不引用string头文件还可以用string
  4. thematic主题框架安装
  5. redis雪崩、穿透、击穿
  6. 6-2 求子串*分数 20
  7. linux所有者和所属组的含义,Linux文件权限(所有者,所属组)及含义详解
  8. 【APP平台化】APP平台化、低代码平台设计思路与实现
  9. 实用拜占庭容错和Raft 算法总结
  10. skip-gram负采样原理