【26 预处理详解】

目录

  1. 预定义符号
  2. #define定义常量
  3. #define定义宏
  4. 带有副作用的宏参数
  5. 宏替换的规则
  6. 宏函数的对比
  7. #和##
  8. 命名约定
  9. #undef
  10. 命令行定义
  11. 条件编译
  12. 头文件的包含
  13. 其他预处理指令

1. 预定义符号

c语言设置了一些预定义符号,可以直接使用,预定义符号也是在预处理期间处理的

FILE //进行编译的源文件
LINE //文件当前的行号
DATE //文件被编译的日期
TIME //文件被编译的时间
STDC //如果编译器遵循ANSI C,其值为1,否则未定义

举个例子:

printf (“file:%s line:%d\n”, FILE, LINE );

2. #define定义常量

基本语法:

#define name stuff

举个例子

#define MAX 1000
#define reg register //为了register这个关键字,创建一个简短的名字
#define do_forever for (;😉 //用更形象的符号来替换一种实现
#define CASE break ; case //在写case语句的时候自动把break写上
//如果定义的 stuff过长,可以分成几行写,除了最后一行外,每行后面加一个反斜杠(续行符),转义回车
#define DEBUG_PRINT printf (“file:%s\t line:%d\t
date:%s\t time:%s\n ,
FILE,LINE,
DATE,TIME”)

思考:在define定义标识符的时候,要不要在最后加上 ;
比如:

#define MAX 1000;
#define MAX 1000

建议不要加上 ;,这样容易导致问题
比如下面的场景:

if (condition)
max = MAX ;
else
max = 0;

如果是加了分号的情况,等替换后,if和else之间就是2条语句,而没有大括号的时候,if后边只能有一条语句,这里会出现语法错误

3. #define定义宏

#define 机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro)

下面是宏的申明方式:
#define name(parament-list) stuff

其中的parament-list是一个由逗号隔开的符号表,它们可能出现在stuff中

注意:

参数列表的左括号必须与name紧邻,如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分

举例:

#define SQUARE( x ) x * x

这个宏接收一个参数x,如果在上述声明之后,你把SQUARE(5);置于程序中,预处理就会用下面这个表达式替换上面的表达式:5 * 5

警告:
这个宏存在一个问题:
观察下面的代码段:

int a = 5 ;
printf(“%d\n”, SQUARE( a + 1 )) ;

乍一看,你可能觉得这段代码将打印36,事实上它将打印11,为什么呢?
替换文本时,参数x被替换成a + 1,所以这条语句实际上变成了:

printf ( “%d\n”,a + 1 * a +1 ) ;

这样就比较清晰了,由替换产生的表达式并没有按照预想的次序进行求值

在宏定义加两个括号,这个问题便轻松的解决了

#define SQUARE(x) (x) * (x)

这也预处理之后就产生了预期的效果:

printf (“%d\n”, (a + 1) * (a + 1) ) ;

这里还有一个宏定义

#define DOUBLE(x) (x) + (x)

定义中我们使用了括号,想避免之前的问题,但是这个宏可能会出现新的错误

int a =5 ;
printf (“%d\n”, 10 * DOUBLE(a)) ;

这将打印什么值呢?看上去,好像打印100,但事实上打印的是55
我们发现替换后:

printf (“%d\n”,10 * (5) + (5)) ;

乘法运算先于宏定义的加法,所以出现了55
这个问题的解决办法就是在宏定义表达式两边加上一对括号就可以了

#define DOUBLE(x) ( (x) + (x) )

提示:

所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或邻近操作符之间不可预料的相互作用

4. 带有副作用的宏参数

当宏参数在宏的定义中出现超过一次的时候,如果有参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果

x + 1 ; //不带副作用
x ++ ; //带有副作用

MAX宏可以证明具有副作用的参数引起的问题

#define MAX(a, b) ( (a) > (b) ? (a) : (b) )

x = 5 ;
y = 8 ;
z = MAX( x++, y++ ) ;
printf(“x=%d y=%d z=%d\n”, x, y, z) ; //输出的结果是什么

这里我们得知道预处理处理之后的结果是什么:

z = ( (x++) > (y++) ? (x++) : (y++)) ;

所以输出的结果是: x = 6 y = 10 z = 9

5. 宏替换的规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤

  1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先被替换
  2. 替换文本随后被插入到程序原本的位置。对于宏,参数名被他们的值所替换
  3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理的过程

注意:

  1. 宏参数和#define定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归
  2. 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被探索

6. 宏函数的对比

宏通常被应用于执行简单的运算

比如在两个数中较大的一个时,写成下面的宏,更有优势一些
#define MAX(a, b) ((a) > (b) ? (a) : (b))

那为什么不用函数来完成这个任务
原因有二:

  1. 用于调用函数和从函数返回代码可能比实际执行这个小型计算工作所需要的时间更多。所以宏比函数在程序的规模和速度方面更胜一筹
  2. 更为重要的是函数的参数必须声明为特点的类型。所以函数只能在类型合适的表达式上使用。反之,这个宏适用于整形、长整型、浮点型等可以用>来比较的类型。宏和类型是无关的

和函数相比宏的优势:

  1. 每次使用宏的时候,一份宏定义的代码插入到程序中,除非宏比较短,否则可能大幅度增加程序的长度
  2. 宏是没法调试的
  3. 宏由于类型无关,也不够严谨
  4. 宏可能会带来运算符优先级的问题,导致容易出错

宏有时候可以做函数做不到的事情。比如:宏的参数可以出现类型,但是函数做不到

#define MALLOC(num, type)
(type)malloc( num sizeof(type) )

//使用
MALLOC(10, int) ; //类型作为参数

//预处理器替换之后
(int)malloc(10 sizeof(int) ) ;

宏和函数的对比:
在这里插入图片描述

7. #和##

7.1 #运算符

#运算符将宏的一个参数转换为字符串面量。它仅允许出现在带参数的宏的替换列表中
#运算符所执行的操作可以理解为“字符串化”
当有一个变量 int a = 10 ;想打印出: the value of a is 10,可以写:

#define PRINT(n) printf (“the value of “#n” is %d”, n) ;

当我们按照下面的方式调用时:
PRINT(a) ;//当我们把a替换到宏的体内时,就出现了#a,而#a就是转换为“a”,字符串代码就会被预处理为:

printf (“the value of “a” is %d”, a) ;

运行代码就能在屏幕上打印:

the value of a is 10

7.2 ##运算符

##可以把位于它两边的符号合成一个符号,它允许宏定义从分离的文本片段创建标识符。##被称为记号粘合

这样的链接必须产生一个合法的标识符。否则结果就是未定义的

这里我们想想,写一个函数求2个数的较大值的时候,不同的数据类型就得写不同的函数

int int_max ( int x, int y )
{
return x > y ? x : y ;
}

float float_max ( float x, float y )
{
return x > y ? x: y ;
}

这样写起来太繁琐了,现在我们这样写代码试试:

//宏定义
#define GENERIC_MAX(type)
type type##_max (type x, type y)
{
return (x > y ? x : y)
}

使用宏,定义不同函数

GENERIC_MAX(int) //替换到宏体内后int##max 生成了新的符号 int_max做函数名
GENERIC_MAX(float) //替换到宏体内后int##max 生成了新的符号 int_max做函数名

int main()
{
//调用函数
int m = int_max(2, 3) ;
printf(“%d\n”, m) ;
float fm = float_max(3.5f, 4.5f) ;
printf(“%f\n”, fm) ;
}

输出:

3
4.500000

8. 命名约定

一般来讲函数的宏使用语法很相似。所以语法本身没法帮我们区分二者

把宏名全部大写
函数名不要全部大写

offsetof并不是函数,本质是宏

9. undef

这条指令用于移除一个宏定义

#undef NAME
//如果现存的一个名字需要被重新定义,那么它的旧名字首先要被移除

10. 命令行定义

许多c的编译器提供了一种能力,允许在命令行中定义符号。用于启动编译过程

例如:当我们根据同一个源文件要编译出一个程序的不同版本时,这个特性有用处。(假定某个程序中声明了一个某个长度的数组,如果机器内存有限,我们需要一个很小的数组,但是另外一个机器内存大些,需要的数组就大些)

int array[ARRAY_SIZE];
int i = 0;
for (i = 0; i < ARRAY_SIZE; i++)
{
	array[i] = i;
}
for (i = 0; i < ARRAY_SIZE; i++)
{
	printf("%d ",array[i]);
}
printf("\n");

linux 环境演示
gcc -D ARRAY_SIZE = 10 programe.c

11. 条件编译

在编译个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便的。因为我们有条件编译指令

比如说:

调试性的代码,删除可惜,保留又碍事,所以我们可以选择性编译

#define __DEBUG__
	int i = 0;
	int arr[10] = { 0 };
	for (i = 0; i < 10; i++)
	{
		arr[i] = i;
#ifdef __DEBUG__
		printf("%d ", arr[i]); //为了观察数组是否赋值成功
#endif  //__DEBUG__
	}

常见的条件编译指令:

#if  常量表达式
//...
#endif
	//常量表达式由预处理器求值
//如:
#define __DEBUG__ 1
#if __DEBUG__
//...
#endif

//2.多个分支的条件编译
#if 常量表达式
//...
#elif 常量表达式
//...
#else
//...
#endif

//3.判断是否被定义
#if defined(symbol)
#endif
#ifdef symbol
#endif

#if !defined(symbol)
#endif
#ifndef symbol
#endif
//4.嵌套指令
#if defined(OS_UNIX)
	#ifdef OPTION1
		unix_version_option1();
	#endif

	#ifdef OPTION2
		unix_version_option2();
	#endif
#elif defined(OS_MSDOS)
	#ifdef OPTION2
		unix_version_option2();
	#endif
#endif

12. 头文件的包含

12.1 头文件被包含的方式:

12.1.1 本地文件包含

include “filename”

查找策略: 先在原文件所在目录查找,如果找不到,像查找库函数一样在环境变量标准位置查找,找不到就错误

linux环境的标准头文件路径:

/usr/include

vs环境的标准头文件路径:

C:\Program Files (x86)\Microsoft Visual Studio 12.0\VC\include
//默认位置

12.1.2 库文件包含

#include <filename.h>

查找头文件直接去标准路径下查找,如果找不到就提示编译错误

这样是不是可以说,对于库文件也可以用“”的形式包含。这个是可以的,不过这样效率就会变低,不容易区分是什么文件

12.2 嵌套文件包含

我们知道,include指令可以使另一个文件被编译,这种替换方式很简单:预处理器先删除这条指令,并用包含文件的内容替换,一个头文件被包含十次,就实际被编译10次,对编译的压力比较大

#include “test.h”
#include “test.h”
#include “test.h”
#include “test.h”
#include “test.h”
#include “test.h”

这样写,就会拷贝6份该头文件

如果这个头文件比较大,处理后的代码量和工程就比较大。如何解决这种问题呢,就是用条件编译

每个头文件开头这样写:

#ifndef TEST_H
#define TEST_H
//头文件内容
#endif //TEST_H

或者

#pragma once

就可以避免头文件重复引入

注:
推荐《高质量C/C++编程指南》中附录的考试试卷
题目:

1.头文件中的ifndef/define/endif是干什么用的?
2.#include <filename.h>和#include "filename,h"有什么区别

13. 其他预处理指令

#error
#pragma
#Line

以上不多做介绍
#pragma pack() //结构体部分

参考《C语言深度解剖》学习

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/319803.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

响应式编程初探-自定义实现Reactive Streams规范

最近在学响应式编程&#xff0c;这里先记录下&#xff0c;响应式编程的一些基础内容 1.名词解释 Reactive Streams、Reactor、WebFlux以及响应式编程之间存在密切的关系&#xff0c;它们共同构成了在Java生态系统中处理异步和响应式编程的一系列工具和框架。 Reactive Streams…

ruoyi后台管理系统部署-1-安装JDK

CentOS 7 安装JDK 1. 最简单方法 部署JDK最简单的方法是使用&#xff1a;yum 首先查看原服务器有没有安装 jdk # 没有输出 java -version yum list installed | grep javayum 安装 查看可供安装的jdk&#xff1a; yum search java | grep jdk yum -y list java*应该输出一堆…

C#无标题栏窗体拖动代码

文章目录 一、概念二、案例三、常见问题四、链接 一、概念 C#&#xff08;C Sharp&#xff09;是由微软公司开发的一种面向对象的编程语言。它是从C和C语言演化而来的&#xff0c;并结合了Java和其他编程语言的特性。C#是微软.NET平台的一部分&#xff0c;允许开发人员创建各种…

超强站群系统v9.0:最新蜘蛛池优化技术,一键安装,内容无缓存刷新,高效安全

安全、高效&#xff0c;化的优化利用php性能&#xff0c;使得运行流畅稳定 独创内容无缓存刷新不变&#xff0c;节省硬盘。防止搜索引擎识别蜘蛛池 蜘蛛池算法&#xff0c;轻松构建站点&#xff08;电影、资讯、图片、论坛等等&#xff09; 可以个性化每个网站的风格、内容、…

YOLOv7涨点改进:多层次特征融合(SDI),小目标涨点明显,| UNet v2,比UNet显存占用更少、参数更少

💡💡💡本文全网独家改进:多层次特征融合(SDI),能够显著提升不同尺度和小目标的识别率 💡💡💡在YOLOv7中如何使用 1)iAFF加入Neck替代Concat; 收录: YOLOv7高阶自研专栏介绍: http://t.csdnimg.cn/tYI0c ✨✨✨前沿最新计算机顶会复现 🚀🚀🚀YOL…

6.2 声音编辑工具GoldWave5简介(7)

6.2.5其它常用功能 1&#xff0e;高低通 把录制的语音和背景音乐融合在一起时&#xff0c;可能会出现背景音乐音量过大&#xff0c;语音音量过小的现象&#xff0c;这时可以选择“低通”将背景音乐的音量降低一些。 (1)选择【效果】|【波波器】|【低通/高通】命令&#xff0…

wireshark使用教程

目录 windows平台安装Wireshark组件选择Additional TasksPacket CaptureUSB CaptureNpcap Installation Options Ubuntu上安装 Wireshark不使用 sudo 运行 Wireshark 使用GUI抓包使用命令行抓包确定抓取哪个网卡的报文抓取数据包停止抓包设置过滤条件 参考资料 Wireshark 是一款…

SpringBoot整合ES

1.引入依赖 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-elasticsearch</artifactId> <version>2.6.3</version> </dependency> 2.config配置文件 Configu…

杨中科 EFCORE 第三部分 主键

主键 自增主键 1、EF Core支持多种主键生成策略:自动增长;Guid;Hi/Lo算法等。 2、自动增长。 优点:简单; 缺点: 数据库迁移以及分布式系统中&#xff08;多数据库合并&#xff0c;会有重复主键值&#xff09;比较麻烦;并发性能差&#xff08;大并发情况下&#xff0c;为了保证…

【LeetCode: 57. 插入区间+分类讨论+模拟】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

NLP论文阅读记录 - WOS | ROUGE-SEM:使用ROUGE结合语义更好地评估摘要

文章目录 前言0、论文摘要一、Introduction1.1目标问题1.2相关的尝试1.3本文贡献 二.相关工作三.本文方法四 实验效果4.1数据集4.2 对比模型4.3实施细节4.4评估指标4.5 实验结果4.6 细粒度分析 五 总结 前言 ROUGE-SEM: Better evaluation of summarization using ROUGE combin…

ssh 远程登录协议

一、SSH 服务 1.1 SSH 基础 SSH&#xff08;Secure Shell&#xff09;是一种安全通道协议&#xff0c;主要用来实现字符界面的远程登录、远程 复制等功能。SSH 协议对通信双方的数据传输进行了加密处理&#xff0c;其中包括用户登录时输入的用户口令&#xff0c;SSH 为建立在应…

STM8入门|第一个工程

开发软件 不支持Keil&#xff0c;使用IAR for STM8&#xff0c;注意 IAR系列有很多种 STM8对应软件是 IAR for STM8 软件下载&#xff1a; 官网下载地址&#xff0c;官网版本下载比较麻烦&#xff0c;可以按教程网盘地址下载。 下载安装教程&#xff1a; https://www.cnblogs…

Compileflow工作流引擎使用讲解

文章目录 1 Compileflow1.1 简介1.2 特点1.3 Compileflow插件下载1.4 main方法调用1.4.1 pom.xml1.4.2 新建bpm文件1.4.3 各个节点绑定方法1.4.4 测试方法 1.5 bpm各个标签说明1.5.1 BPM根节点1.5.2 全局变量1.5.3 开始节点: start1.5.4 结束节点: end1.5.5 自动节点: autoTask…

B-TREE(B-树)

B-TREE B-tree 又叫平衡多路查找树。一棵 m 阶的 B-tree (m 叉树)的特性如下&#xff08;其中 ceil(x)是一个取上限的函数&#xff09;&#xff1a; 树中每个结点至多有 m 个孩子&#xff1b; 除根结点和叶子结点外&#xff0c;其它每个结点至少有有 ceil(m / 2)个孩子&#…

JVM实战(13)——JVM优化概述

作者简介&#xff1a;大家好&#xff0c;我是smart哥&#xff0c;前中兴通讯、美团架构师&#xff0c;现某互联网公司CTO 联系qq&#xff1a;184480602&#xff0c;加我进群&#xff0c;大家一起学习&#xff0c;一起进步&#xff0c;一起对抗互联网寒冬 学习必须往深处挖&…

每日一题——LeetCode1128.等价多米诺骨牌对的数量

先尝试暴力解法&#xff1a; var numEquivDominoPairs function(dominoes) {var count0for(let i0;i<dominoes.length-1;i){for(let ji1;j<dominoes.length;j){if((dominoes[i][0]dominoes[j][0] && dominoes[i][1]dominoes[j][1]) || (dominoes[i][0]dominoes…

Qt/QML编程学习之心得:小键盘keyboard(36)

小键盘对于qml应用是经常用到的,在qml里面,就如一个fileDialog也要自己画一样,小键盘keyboard也是要自己画的,对于相应的每个按键的clicked都要一一实现的。 这里有一个示例: 代码如下: import QtQuick 2.5 import QtQuick.Controls 1.4 import QtQuick.Window 2.0 im…

Redis面试篇

redis面试题主要内容 面试官在面试时主要会问以下这些方面的问题 下面是一些问题示例&#xff1a; redis-使用场景 缓存 缓存穿透 介绍 缓存穿透&#xff1a;查询一个不存在的数据&#xff0c;mysql查询不到数据也不会直接写入缓存&#xff0c;就会导致每次请求都会去查数…

2.1 常用计算机网络体系结构

2.1 常用计算机网络体系结构 2.1.1 OSI体系结构 1、为了使不同体系结构的计算机网络都能够互联&#xff0c;国际标准化组织于1977年成立了专门机构研究该问题&#xff0c;不久他们就提出了一个试图使各种计算机在世界范围内都能够互连成网的标准框架&#xff0c;也就是著名的…
最新文章