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 "<command line>"
# 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 "<command line>"
# 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)

你可能还喜欢下面这些文章

redis的RDB文件存储结构分析

原文标题:15天玩转redis —— 第十一篇 让你彻底了解RDB存储结构这里我们来继续分析一下RDB文件存储结构,首先大家都知道RDB文件是在redis的“快照”的模式下才会产生,那么如果我们理解了RDB文件的结构,是不是让我们对“快照”模式能做到一个心中有数呢?一:RDB结构剖析首先呢,我们要对RDB文件有一个概念性的认识,比如下面画的图一样: 从图中,我们大概看到了RDB文件的一个简要的存储模式,但为了更好的方便对照,我准备save一个empty database,对比一下看看效果: 然后我们用winHex打开dump.rdb文件,看看它的16进制。好了,该打开的我都

Go入门:四、面向对象

这是我的Go学习笔记的第四篇,面向对象!现代语言几乎都会面向对象进行了支持!当然,Go也具备面向对象的特性!我的语言学习过程一般分为下面几个:1. 变量和数据类型2. 流程控制方法3. 函数声明和调用4. 面向对象5. 语言特性6. 标准库Go语言中的面向对象有点特殊。在Go语言里面,没有显式的class、extends等面向对象语言经常使用的关键词,但是却有面向对象的特性。看看Go怎么实现的把!创建一个类按照我的理解,类实际上就是某种模板,这个模板里面含有有限多个属性和方法。在Go里面,定义这个模板的语法使用type来实现!比如单个int类型可以构成一个类(没错,你甚至可以在int数据类型上

如何选择特征

特征工程是数据分析中最耗时间和精力的一部分工作,它不像算法和模型那样是确定的步骤,更多是工程上的经验和权衡。因此没有统一的方法。这里只是对一些常用的方法做一个总结。本文关注于特征选择部分。后面还有两篇会关注于特征表达和特征预处理。1. 特征的来源在做数据分析的时候,特征的来源一般有两块,一块是业务已经整理好各种特征数据,我们需要去找出适合我们问题需要的特征;另一块是我们从业务特征中自己去寻找高级数据特征。我们就针对这两部分来分别讨论。2.  选择合适的特征我们首先看当业务已经整理好各种特征数据时,我们如何去找出适合我们问题需要的特征,此时特征数可能成百上千,哪些才是我们需要的呢?第一

记一次进程异常退出的问题排查

机器搬家之后,之前一直稳定的PHP多进程程序子进程突然异常退出,但是退出的不是很频繁,查看进程日志并也没有发现有什么导致退出的,问题比较诡异。于是开启了一段问题排查之路。首先查看内核日志,使用dmesg,拉到最后发现有一些这样的错误,看来确实是崩溃了。 php: segfault at 7f6443ee18c8 ip 00007f6443ee18c8 sp 00007fff4d4ba818 error 15 in libc-2.17.so php: segfault at 0 ip 000000000075919d sp 00007fff0c6e0578 error 4 in php trap

matplotlib画柱状图教程

matplotlib画柱状图十分简单,首先需要导入matplotlib中的pyplot,这个就是能够画图的类啦import matplotlib.pyplot as plt画柱状图十分简单,使用bar方法,比如画一个各个班级的人数吧,先准备一些数据,有两组数据,students为人数,这里放到y坐标系中,变量x就放在x坐标系中students = x = 好了,现在我们可以直接画出来,调用bar方法,最后再调用show就能画图啦,代码如下plt.bar(x=x, height=students)plt.show()完整代码如下#coding:utf-8import matplotlib.pyp

布隆过滤器(bloom filter)介绍以及php和redis实现布隆过滤器实现方法

引言在介绍布隆过滤器之前我们首先引入几个场景。场景一在一个高并发的计数系统中,如果一个key没有计数,此时我们应该返回0。但是访问的key不存在,相当于每次访问缓存都不起作用了。那么如何避免频繁访问数量为0的key而导致的缓存被击穿?有人说, 将这个key的值置为0存入缓存不就行了吗?这是确实是一种解决方案。当访问一个不存在的key的时候,设置一个带有过期时间的标志,然后放入缓存。不过这样做的缺点也很明显:浪费内存和无法抵御随机key攻击。场景二在一个黑名单系统中,我们需要设置很多黑名单内容。比如一个邮件系统,我们需要设置黑名单用户,当判断垃圾邮件的时候,要怎么去做。比如爬虫系统,我们要记录下

php的empty,isset,is_null与!

来说说php的empty,isset,is_null 与!,这几个都是if语句中比较常见的判断逻辑。但是有时候用的很纠结,甚至看别人写的程序里面也很纠结。特地梳理梳理,避免踩坑先来定义一些东西<?php$a;$b=0;$c=array();$d='';$e=null;empty,用了会上瘾这是一个用了会上瘾的语言结构!多好,empty可接受的参数是一个变量,任意类型,哪怕是变量不存在,只要变量被boolean转换之后是false(参考:php的boolean都有哪些),那么empty返回的就是false,并且不会出现警告!等价于不过注意的是,empty里面不能使用表达式(在php<

gcc/g++编译参数详解

编译步骤gcc 与 g++ 分别是 gnu 的 c & c++ 编译器。gcc/g++ 在执行编译工作的时候,总共需要4步:预处理,生成 .i 的文件将预处理后的文件转换成汇编语言, 生成文件 .s 有汇编变为目标代码(机器代码)生成 .o 的文件连接目标代码, 生成可执行程序 参数详解-x language filename参数含义为指定文件所使用的语言。根据约定,C语言的后缀名称为".c",而 C++ 的后缀名为".cpp"或".cc",但如果你的源代码后缀不约定的那几种,那么需要使用-x参数来指定文件所使用的语言。这个参数对他后面的文件名都起作用。 可以使用的参数吗有下面的这些:

创建自己的composer包之怎样制作composer包

项目做多了之后,可能会慢慢总结出自己的代码库出来,当在新的项目中使用的时候,总不能一直是复制粘贴。这个时候,composer就能派上用场了。一个项目开始,使用composer就能够加载所需要的依赖,非常方便。这个时候,来做一个自己的包吧!使用命名空间composer自动加载需要用到命名空间,因此所有的代码库都需要使用命名空间,如果没有,那就改吧!使用命名空间之后你可能会打开新世界的大门。创建composer.json我假设你已经安装了composer,并且已经会使用了。创建自己的包我们首先需要创建一个composer.json,示例文件如下上面的composer.json有一个比较重要的是au

linux shell 入门

从程序员的角度来看, Shell本身是一种用C语言编写的程序,从用户的角度来看,Shell是用户与Linux操作系统沟通的桥梁。用户既可以输入命令执行,又可以利用 Shell脚本编程,完成更加复杂的操作。在Linux GUI日益完善的今天,在系统管理等领域,Shell编程仍然起着不可忽视的作用。深入地了解和熟练地掌握Shell编程,是每一个Linux用户的必修 功课之一。Linux的Shell种类众多,常见的有:Bourne Shell(/usr/bin/sh或/bin/sh)、Bourne Again Shell(/bin/bash)、C Shell(/usr/bin/csh)、K Shel

赞赏

微信赞赏支付宝赞赏

发表回复

您的电子邮箱地址不会被公开。 必填项已用 * 标注