详解C语言#define预处理宏定义

目录
  • #define介绍:
  • #define宏定义无参的一般形式为:#define  标识符 常量
  • #define宏定义有参的一般形式为:#define  标识符(参数表) 表达式
  • #运算符:
  • ##运算符:
  • 可变宏...和__VA_ARGS__:
  • 开发项目中常用的宏定义:

#define介绍:

C语言里可以用#define定义一个标识符来表示一个常量。特点是:定义的标识符不占内存,只是一个临时的符号,预编译后这个符号就不存在了,也不做类型定义。预编译又叫预处理。预编译就是编译前的处理。这个操作是在正式编译之前由系统自动完成的。

#define又称宏定义,标识符为所定义的宏名,简称宏。标识符的命名规则和变量的命名规则是一样的。#define的功能是将标识符定义为其后的常量,一经定义,程序中就可以直接用标识符来表示这个常量,也就是文本替换。变量名表示的是一个变量,但宏名表示的是一个常量,可以给变量赋值,但绝不能给常量赋值。

宏定义最大的好处是方便程序的修改。使用宏定义可以用宏代替一个在程序中经常使用的常量。这样,当需要改变这个常量的值时,就不需要对整个程序一个一个进行修改,只需修改宏定义中的常量就行了。且当常量比较长时,使用宏就可以用较短的有意义的标识符来代替它,这样编程的时候就会更方便,不容易出错。因此,宏定义的优点就是方便和易于维护。

#define宏定义无参的一般形式为:#define  标识符 常量

注意最后没有分号,因为宏不是语句,结尾不用加分号,否则会被替换进进程中。还有一点就是宏名最好用大写字母加下划线组成,以此来区分变量名。

来看一个#define宏定义无参的例子:

#include<stdio.h>
#define PI 3.1415926//标识符或宏名叫PI 常量是个浮点型 作用是圆周率
#define R 2//标识符或宏名叫R 常量是个整型 作用是圆的半径
#define PRINT "半经为2的圆 面积=%lf\n"//标识符或宏名叫PRINT 常量是个字符串 作用是代替了printf()函数的第一个参数
int main()
{
    printf(PRINT,PI*R*R);//在这里PRINT被替换成"半经为2的圆 面积=%lf" PI被替换成3.1415926 R被替换成2
    printf("半经为2的圆 面积=%lf\n",3.1415926*2*2);//这句是上面一句代码替换后的代码
    return 0;
}

#define宏定义有参的一般形式为:#define  标识符(参数表) 表达式

带参数的宏定义,宏名中不能有空格,宏名与形参表之间也不能有空格,而形参表中形参之间可以有空格。

来看一个#define宏定义有参的例子:

#include<stdio.h>
#define SQUARE(x) x*x//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",SQUARE(3),SQUARE(4));//SQUARE(3)被替换成3*3 SQUARE(4)被替换成4*4
 printf("%d %d\n",3*3,4*4);//这句是上面一句代码替换后的代码
 return 0;
}

我门来稍微改动下代码:

#include<stdio.h>
#define SQUARE(x) x*x//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",SQUARE(2+1),SQUARE(3+1));
 return 0;
}

这里只是把传的参数3被改成了2+1,4被改成了3+1,可能有些朋友会说2+1=3,3+1=4,答案不就和刚才一样嘛。其实不是,因为#define宏定义只是简单的文本替换,那应该被替换成什么呢。SQUARE(2+1)和SQUARE(3+1)分别替换成2+1*2+1和3+1*3+1, 那么最终答案自然是5和7了。那么如何杜绝这个问题呢?

很简单,只要在传参时多加一层小括号:

#include<stdio.h>
#define SQUARE(x) x*x//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",SQUARE((2+1)),SQUARE((3+1)));//SQUARE((2+1))被替换成(2+1)*(2+1) SQUARE((3+1))被替换成(3+1)*(3+1)
 printf("%d %d\n",(2+1)*(2+1),(3+1)*(3+1));//这句是上面一句代码替换后的代码
 return 0;
}

如果觉得这样太繁杂,麻烦了,那么直接在宏定义的表达式那里的每个参数都加上小括号:

#include<stdio.h>
#define SQUARE(x) (x)*(x)//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",SQUARE(2+1),SQUARE(3+1));//SQUARE((2+1))被替换成(2+1)*(2+1) SQUARE((3+1))被替换成(3+1)*(3+1)
 printf("%d %d\n",(2+1)*(2+1),(3+1)*(3+1));//这句是上面一句代码替换后的代码
 return 0;
}

我们又来稍微改动下代码:

#include<stdio.h>
#define SQUARE(x) (x)*(x)//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",9/SQUARE(3),16/SQUARE(4));
 return 0;
}

这里在传参数之前加了个除法,那么3*3后是9,再被9除,等于1,4*4后是16,再被16除,等于1,那么预想的最终答案就是1和1了,其实也不是。再来一次文本替换,9/SQUARE(3)和16/SQUARE(4)分别替换成9/(3)*(3)和16/(4)*(4), 那么最终答案自然是9和16了。那么又如何杜绝这个问题呢?

也很简单,只要在传参时多加一层小括号:

#include<stdio.h>
#define SQUARE(x) (x)*(x)//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",9/(SQUARE(3)),16/(SQUARE(4)));//9/(SQUARE(3))和16/(SQUARE(4))分别替换成9/((3)*(3))和16/((4)*(4))
 printf("%d %d\n",9/((3)*(3)),16/((4)*(4)));//这句是上面一句代码替换后的代码
 return 0;
}

如果依然觉得这样太繁杂,麻烦了,那么直接在宏定义的表达式那里的整个表达式都加上小括号:

#include<stdio.h>
#define SQUARE(x) ((x)*(x))//标识符或宏名叫SQUARE 表达式是x*x 作用是算x的平方
int main()
{
 printf("%d %d\n",9/SQUARE(3),16/SQUARE(4));//9/(SQUARE(3))和16/(SQUARE(4))分别替换成9/((3)*(3))和16/((4)*(4))
 printf("%d %d\n",9/((3)*(3)),16/((4)*(4)));//这句是上面一句代码替换后的代码
 return 0;
}

表达式也可以写多个语句:

#include<stdio.h>
#define AB(a,b) a=i+5,b=j+3
int main()
{
    int i=3,j=5,m=0,n=0;
    AB(m,n);//AB(m,n)被替换成m=i+5,n=j+3
    printf("%d %d\n",m,n);
    return 0;
}

#运算符:

#运算符的作用就是将#后边的宏参数进行字符串的操作,也就是将#后边的参数两边加上一对双引号使其成为字符串。例如param是一个宏的形参,则替换文本中的#param被系统转化为"param",这个转换过程即为字符串化。如下代码:

#include<stdio.h>
#define TEST(param) #param//标识符或宏名叫TEST 表达式是#param 作用是把param参数转换为字符串
int main()
{
    printf("%s\n",TEST(换行前\n第一次换行\n第二次换行));//TEST(换行前\n第一次换行\n第二次换行)被替换成"换行前\n第一次换行\n第二次换行"
    printf("换行前\n第一次换行\n第二次换行\n");//这句是上面一句代码替换后的代码
 return 0;
}

##运算符:

##运算符也可以用在替换文本中,它的作用起到粘合的作用,即将两个宏参数连接为一个数。如下代码:

#include<stdio.h>
#define TEST(param1,param2) (param1##param2)//标识符或宏名叫TEST 表达式是(param1##param2) 作用是把param1参数和param2参数和连接为一个数
int main()
{
    printf("%d\n",TEST(12,34));//TEST(12,34)被替换成(1234)
    printf("%d\n",(1234));//这句是上面一句代码替换后的代码
    return 0;
}

可变宏...和__VA_ARGS__:

可变宏...和__VA_ARGS__的作用主要是为了方便管理软件中的打印信息。在写代码或DEBUG时通常需要将一些重要参数打印出来,但在软件发行的时候不希望有这些打印,这时就用到可变参数宏了。如下代码:

#include<stdio.h>
#define PRINT(...) printf(__VA_ARGS__)//标识符或宏名叫PRINT 表达式是printf(__VA_ARGS__) __VA_ARGS__被用在替换文本中,来表示省略号...代表了什么
int main()
{
    PRINT("hello\n");//PRINT("hello\n")被替换成printf("hello\n")
    printf("hello\n");//这句是上面一句代码替换后的代码
    return 0;
}

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

开发项目中常用的宏定义:

防止头文件被重复包含:

#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]))

到此这篇关于详解C语言#define预处理宏定义 的文章就介绍到这了,更多相关C语言#define预处理宏定义 内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • 详解C语言中的#define宏定义命令用法

    #define 命令#define定义了一个标识符及一个串.在源程序中每次遇到该标识符时,均以定义的串代换它.ANSI标准将标识符定义为宏名,将替换过程称为宏替换.命令的一般形式为: #define identifier string 注意: 1.该语句没有分号.在标识符和串之间可以有任意个空格,串一旦开始,仅由一新行结束. 2.宏名定义后,即可成为其它宏名定义中的一部分. 3.宏替换仅仅是以文本串代替宏标识符,前提是宏标识符必须独立的识别出来,否则不进行替换.例如: #define XYZ t

  • C语言中typedef的用法以及#define区别详解

    目录 1.简洁定义 2.为已有类型起别名 为字符数组起别名 为指针起别名 3.typedef 和 #define 的区别 总结 1.简洁定义 C语言允许为一个数据类型起一个新的别名,就像给人起"绰号"一样.而编程中起别名,是为了编程人员编程方便,例如: 定义如下结构体 struct stu { int ID; char name[20]; float score[3]; char *data; }; 要想定义一个结构体变量就得这样写: struct stu Marry://Marry是

  • C语言中的内联函数(inline)与宏定义(#define)详细解析

    先简明扼要,说下关键:1.内联函数在可读性方面与函数是相同的,而在编译时是将函数直接嵌入调用程序的主体,省去了调用/返回指令,这样在运行时速度更快. 2.内联函数可以调试,而宏定义是不可以调试的.内联函数与宏本质上是两个不同的概念如果程序编写者对于既要求快速,又要求可读的情况下,则应该将函数冠以inline.下面详细介绍一下探讨一下内联函数与宏定义. 一.内联函数是什么?内联函数是代码被插入到调用者代码处的函数.如同 #define 宏(但并不等同,原因见下文),内联函数通过避免被调用的开销来提

  • C语言入门篇--定义宏#define的概述

    目录 1.什么是宏 2.为什么要有宏? 3.宏的用法 3.1定义宏 3.2宏常量 3.3"宏函数" 1.什么是宏 宏只做简单的文本替换,但一般在作为变量名出现或者在字符串中出现中是不会被替换的. 2.为什么要有宏? (1)便于代码的可维护性,对宏的内容进行修改,即可修改代码中所有用到此宏地方的内容. (2)方便阅读,见名知意. 3.宏的用法 3.1定义宏 格式: #define 标识符 内容 3.2宏常量 #include <stdio.h> #define M 1 int

  • C语言中#define与typedef的互换细节详解

    复制代码 代码如下: #include <stdio.h>/*<---------           #define    string    char *            ---->*/typedef   char *   string; int main(void){   string   a[] = {"I", "like", "to", "fight,"},   b[] = {"

  • C语言编程技巧 关于const和#define的区别心得

    #define ASPECT_RATIO 1.653 编译器会永远也看不到ASPECT_RATIO这个符号名,因为在源码进入编译器之前,它会被预处理程序去掉,于是ASPECT_RATIO不会加入到符号列表中.如果涉及到这个常量的代码在编译时报错,就会很令人费解,因为报错信息指的是1.653,而不是ASPECT_RATIO.如果ASPECT_RATIO不是在你自己写的头文件中定义的,你就会奇怪1.653是从哪里来的,甚至会花时间跟踪下去.这个问题也会出现在符号调试器中,因为同样地,你所写的符号名不

  • 浅谈关于C语言中#define的副作用

    MFC虽然没有未来,但是我觉得MFC的思想还是有必要研究研究的,在MFC中或者一些底层代码的编写中,宏是相当好用的,为什么呢?因为宏只是简单的替换,不进行类型转换,替换就意味着灵活,而C语言编程的灵魂就是灵活啊. 但是在高级语言中,甚至C++中,是提倡用const的,不提倡用#define,因为#define有一定的副作用,玩不好就没命了. 这里我们讨论下#define的副作用. # include <stdio.h> # include <stdlib.h> # define C

  • C语言#define拼接宏定义实现方式

    使用场合:拼接两个宏,一个是传入的宏.但是传入的宏不会被替换,反而原封不动的接了上去,这就尴尬了.经过各种尝试,居然成了,特此记录分享一下,方便大家学习. char A_param=0; char B_pramm=0; //添加宏定义 #define OBJECT A #define DEFINE_(X) X##_param //一次定义 #define DEFINE(X) DEFINE_(X) //再次定义 #define PARAM DEFINE(OBJECT) void fun() { /

  • 详解C语言#define预处理宏定义

    目录 #define介绍: #define宏定义无参的一般形式为:#define  标识符 常量 #define宏定义有参的一般形式为:#define  标识符(参数表) 表达式 #运算符: ##运算符: 可变宏...和__VA_ARGS__: 开发项目中常用的宏定义: #define介绍: C语言里可以用#define定义一个标识符来表示一个常量.特点是:定义的标识符不占内存,只是一个临时的符号,预编译后这个符号就不存在了,也不做类型定义.预编译又叫预处理.预编译就是编译前的处理.这个操作是在

  • 详解C语言的预处理效果

    目录 前言 一.预定义符号 二.#define 1.宏 2.宏与函数 3.带副作用的宏参数 4. 宏和函数的不同 5.#undef 三.条件编译 四.文件包含 1.函数库文件包含 2.本地文件包含 总结 前言 编译一个C语言程序涉及很多步骤.其中第一个步骤被称为预处理.C语言的预处理器在源代码编译之前对其进行一些文本性质的操作.它的主要任务包括删除注释.插入被#include指令包含的文件内容.定义和替换由#define指令定义的符号,同时确定代码的部分内容是否应该根据一些条件编译指令进行编译.

  • 详解C语言之预处理(下)

    目录 #define定义宏带副作用的宏参数 #define定义宏的优点 #define定义宏劣势 预处理 预定义符号 预处理指令 条件编译 1.调试性代码 2.防止重复的头文件多次编译 总结 #define定义宏带副作用的宏参数 我们来看如下一段代码 结果分别为12,11,13 当参数替换后,首先判断表达式 (a++)>(b++)?,判断后a的值加1 b的值加1,然后执行表达式(b++)此时执行的值为12,执行完成后b的值加1,则a的值为11,b的值为13.可以看出对于这种情况下的宏是带有副作用

  • 详解C语言之预处理

    目录 程序的翻译环境 编译 预编译: 编译: 汇编: 链接 合并段表: #define的用法 1.#define定义标识符,例如 2.#define定义宏 3.#define实现将参数插入到字符串中 总结 程序的翻译环境 源文件被转换成可执行的机器指令时所处的环境称为翻译环境. 由源文件(.c)转换成可执行文件(.exe)需要两步 编译通过编译器实现,链接通过链接器实现 每个源文件都会经过编译器处理后生成对应的目标文件,然后链接器将目标文件和链接库链接在一起生成可执行程序 编译和链接的具体操作

  • 详解C语言之预处理(上)

    目录 程序的翻译环境 编译 预编译: 编译: 汇编: 链接 合并段表: #define的用法 1.#define定义标识符,例如 2.#define定义宏 3.#define实现将参数插入到字符串中 总结 程序的翻译环境 源文件被转换成可执行的机器指令时所处的环境称为翻译环境. 由源文件(.c)转换成可执行文件(.exe)需要两步 编译通过编译器实现,链接通过链接器实现 每个源文件都会经过编译器处理后生成对应的目标文件,然后链接器将目标文件和链接库链接在一起生成可执行程序 编译和链接的具体操作

  • 详解C语言结构体的定义和使用

    目录 1.1: 结构体用来干嘛? 1.2:结构体变量的基本定义格式 1.3:结构体变量的定义 1.4结构体变量的三种引用方法 2.结构体变量的使用(直接使用结构体变量) 2.1输出结果 3.使用结构体指针操作,返回总成绩最低的学生信息 3.1运行结果 总结: 1.1: 结构体用来干嘛? 生活中我们会遇到很多的表格,就比如你的学习成绩表,有姓名 ,学号,各科的成绩,总的成绩等等,这是一些不同的数据类型,我们要是在c语言中想同时使用这些不同的数据怎么办呢? 可以使用结构体变量,结构体变量在c语言中是

  • C语言详细分析宏定义与预处理命令的应用

    目录 宏定义与预处理命令 预处理命令 - 宏定义 定义符号常量 定义傻瓜表达式 定义代码段 预定义的宏 函数 VS 宏定义 预处理命令 - 条件式编译 示例 宏定义与预处理命令 预处理阶段:处理宏定义与预处理命令: 编译期:检查代码,分析语法.语义等,最后生成.o或.obj文件: 链接期:链接所有的.o或.obj文件,生成可执行文件. 预处理命令 - 宏定义 定义符号常量 #define PI 3.1415926 #define MAX_N 10000 定义傻瓜表达式 #define MAX(a

  • 一文带你搞懂C语言预处理宏定义

    目录 预定义符号 #define #define 定义标识符 #define 定义宏 替换规则 # 和## 预定义符号 这些预定义符号都是语言内置的 __FILE__ //进行编译的源文件 __LINE__ //文件当前的行号 __DATE__ //文件被编译的日期 __TIME__ //文件被编译的时间 __STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义 VS环境下未定义__STDC__ ,说明Visual Studio并未完全遵循ANSI C. #define #defi

  • 详解C 语言项目中.h文件和.c文件的关系

    详解C 语言项目中.h文件和.c文件的关系 在编译器只认识.c(.cpp))文件,而不知道.h是何物的年代,那时的人们写了很多的.c(.cpp)文件,渐渐地,人们发现在很多.c(.cpp)文件中的声明语句就是相同的,但他们却不得不一个字一个字地重复地将这些内容敲入每个.c(.cpp)文件.但更为恐怖的是,当其中一个声明有变更时,就需要检查所有的.c(.cpp)文件. 于是人们将重复的部分提取出来,放在一个新文件里,然后在需要的.c(.cpp)文件中敲入#include XXXX这样的语句.这样即

随机推荐