百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

详解C语言中的#define、#undef、#indef、#ifndef#if,#elif等

wptr33 2025-02-26 14:06 20 浏览

1. 明示常量#define

#define为C语言的一个预处理指令,通常用于进行宏定义。每行#define(逻辑行)一般由以下三部分组成,第一部分是#define指令本身,第二部分为宏,第三部分为称为替换列表或替换体

预处理器在发现程序中的宏后,会用宏等价的替换体进行替换,如在上图中,LENGTH 将被替换为100。但值得注意的是双引号中的宏将不会进行替换。来看下面的一个例子:

输出的结果将是:

2. 在define中使用参数

在#define中还可以使用参数创建作用与函数类似的类函数宏。带有参数的宏看上去很像函数,因为这样的宏也使用圆括号。类函数宏定义的圆括号中可以有一个或多个参数,随后这些参数出现在替换体中,如下图所示:

首先预处理器将所有出现MEAN(X,Y)的地方都替换为(((X)+(Y))/2),然后根据X,Y的值进行计算(注意预处理器不做计算,不求值,只替换字符序列)。下面我们来看一个例子:

输出结果为:

前面两行的结果大家应该都能想到,后面两行有部分读者可能会不太明白。还记得我们在上面谈到的嘛,“首先预处理器将所有出现SQUARE(X)的地方都替换为X*X”。对SQUARE(x+2),将其替换为x+2*x+2,x值为5,由于乘号优先级高于+号,所以结果为17,而非49,要得到正确结果我们应将宏定义写为:

对100/SQUARE(2),首先将其替换为100/2*2,根据优先级规则,从左往右对表达式求值,结果为100,而非25,要得到宏定义结果我们应将宏定义写为:

要处理前面的两种情况,应这样定义:

尽管如此,仍无法 避免类似SQUARE(++x)的情况,这里不再深入讨论

在上面我们可以看到类函数宏虽然和函数调用看上去相似,但行为却并不相同,函数调用在程序运行时把参数的值传递给函数,宏调用在编译之前把参数记号传递给程序,这两个不同的过程发生在不同的时期。(务必记得预处理器不做计算,不求值,只替换字符序列

2.2 用宏参数创建字符串:#运算符

我们在前面提到双引号字符串中的宏不会被替换,那么如果我们想要在字符串中包含宏参数该如何做呢?在类函数宏的替换体中,#号作为一个预处理运算符,可以把记号转换为字符串,如#X将被转换为”X"。来看一个例子:

输出结果为:

为什么呢?调用第一个宏时,X在双引号中不会被替换,仅替换((X)*(X));调用第二个宏时,#X将被替换为"X",然后由于字符串的串联特性,"X"将与"The square of "和“is %d\n"组合成"The square of X is %d\n"

2.3 预处理器粘合剂:##运算符

##运算符把两个记号组合成一个记号,如:

调用XNAME(n)将转换为xn,例:

结果为:

2.4 变参宏:...和__VA_ARGS__

通过把宏参数列表中最后的参数写成省略号(...)来实现宏参数可变,而__VA_ARGS__则出现在替换部分中,表明省略号代表什么,如:

结果为:

注意:省略号只能代替最后的宏参数,像下面这样就是不行的

3. undef指令

#undef指令用于”取消“已定义的#define指令。

假如有如下定义:

通过

将移除上面的定义,然后即可将LENGTH定义为一个新值。即使原来没有定义LENGTH,取消LENGTH的定义仍然有效。如果想使用一个名称,又不确定之前是否已经用过,可以用#undef指令取消该名字的定义

4. 条件编译

4.1 #ifdef、#else和#endif指令

#ifdef指令表示如果预处理器已定义了后面的标识符LENGTH_H,则执行#else(如果有)、#endif指令之前的所有指令并编译C代码,如果预处理器未定义标识符LENGTH_H,且有#else指令,则执行#else和#endif指令之间的所有代码

注意:#else可以没有,但#endif必须存在

4.2 #ifndef指令

#ifndef指令和#ifdef指令的逻辑相反,#ifndef指令判断后面的标识符是否是未定义的,常用于定义之前未定义的常量,如:

#ifndef指令也可以和#else、#endif一起使用

通常,包含多个头文件时,其他的文件可能包含相同的宏定义,#ifndef指令可以防止相同的宏被重复定义。在首次定义一个宏的头文件中用#ifndef指令激活定义,随后在其他头文件中的定义都被忽略

#ifndef指令还有一个非常重要的用法,防止多次包含一个文件,读者也许见过这样的写法:

这样写是为什么呢?

首先STACK_H是一个空宏,假如该文件被包含多次,当预处理器首次发现该文件被包含时,STACK_H是未定义的,所以定义STACK_H,并处理该文件的 其他部分,当预处理器第二次发现该文件被包含时,STACK_H是已定义的,预处理器跳过该文件的其他部分

为什么会多次包含一个文件呢,因为在大型程序中,许多被包含的文件中都包含着其他文件,所以显示包含的文件中可能包含着已经包含的其他文件。因为在被包含的文件中有某些项(如一些结构类型的声明)只能在一个文件中出现一次,这样就会出错

通过#ifndef就可以避免重复,因为#ifndef和#endif之间的其他部分在第二次时不会再处理

如何保证像STACK_H这样待测试的标识符没有在别处定义呢?通常可以用用大写的文件名及下划线和大写的H做标识符,如STACK就是文件名stack的大写

(感兴趣的读者可以去看一下我在这篇文章中提的一个关于#ifndef的问题:

关于全局变量被定义在一个被多个.c文件包含的头文件时出现错误

#if和#elif指令

#if指令和if很像,#if后面跟整型常量表达式,如果表达式非零,则表达式为真,此外可以按照if else的形式使用#elif

如:

#if还有一种用法可以代替#ifdef,即#if defined (VAR)代替#ifdef VAR

#defined是一个预处理运算符(注意不要和#define搞混),如果它的参数是用#define定义过的,返回1,否则返回0,这种方法还可以和#elif一起使用

最后觉得这篇文章对你有帮助的读者给个点赞加关注吧!

相关推荐

MySQL进阶五之自动读写分离mysql-proxy

自动读写分离目前,大量现网用户的业务场景中存在读多写少、业务负载无法预测等情况,在有大量读请求的应用场景下,单个实例可能无法承受读取压力,甚至会对业务产生影响。为了实现读取能力的弹性扩展,分担数据库压...

Postgres vs MySQL_vs2022连接mysql数据库

...

3分钟短文 | Laravel SQL筛选两个日期之间的记录,怎么写?

引言今天说一个细分的需求,在模型中,或者使用laravel提供的EloquentORM功能,构造查询语句时,返回位于两个指定的日期之间的条目。应该怎么写?本文通过几个例子,为大家梳理一下。学习时...

一文由浅入深带你完全掌握MySQL的锁机制原理与应用

本文将跟大家聊聊InnoDB的锁。本文比较长,包括一条SQL是如何加锁的,一些加锁规则、如何分析和解决死锁问题等内容,建议耐心读完,肯定对大家有帮助的。为什么需要加锁呢?...

验证Mysql中联合索引的最左匹配原则

后端面试中一定是必问mysql的,在以往的面试中好几个面试官都反馈我Mysql基础不行,今天来着重复习一下自己的弱点知识。在Mysql调优中索引优化又是非常重要的方法,不管公司的大小只要后端项目中用到...

MySQL索引解析(联合索引/最左前缀/覆盖索引/索引下推)

目录1.索引基础...

你会看 MySQL 的执行计划(EXPLAIN)吗?

SQL执行太慢怎么办?我们通常会使用EXPLAIN命令来查看SQL的执行计划,然后根据执行计划找出问题所在并进行优化。用法简介...

MySQL 从入门到精通(四)之索引结构

索引概述索引(index),是帮助MySQL高效获取数据的数据结构(有序),在数据之外,数据库系统还维护者满足特定查询算法的数据结构,这些数据结构以某种方式引用(指向)数据,这样就可以在这些数据结构...

mysql总结——面试中最常问到的知识点

mysql作为开源数据库中的榜一大哥,一直是面试官们考察的重中之重。今天,我们来总结一下mysql的知识点,供大家复习参照,看完这些知识点,再加上一些边角细节,基本上能够应付大多mysql相关面试了(...

mysql总结——面试中最常问到的知识点(2)

首先我们回顾一下上篇内容,主要复习了索引,事务,锁,以及SQL优化的工具。本篇文章接着写后面的内容。性能优化索引优化,SQL中索引的相关优化主要有以下几个方面:最好是全匹配。如果是联合索引的话,遵循最...

MySQL基础全知全解!超详细无废话!轻松上手~

本期内容提醒:全篇2300+字,篇幅较长,可搭配饭菜一同“食”用,全篇无废话(除了这句),干货满满,可收藏供后期反复观看。注:MySQL中语法不区分大小写,本篇中...

深入剖析 MySQL 中的锁机制原理_mysql 锁详解

在互联网软件开发领域,MySQL作为一款广泛应用的关系型数据库管理系统,其锁机制在保障数据一致性和实现并发控制方面扮演着举足轻重的角色。对于互联网软件开发人员而言,深入理解MySQL的锁机制原理...

Java 与 MySQL 性能优化:MySQL分区表设计与性能优化全解析

引言在数据库管理领域,随着数据量的不断增长,如何高效地管理和操作数据成为了一个关键问题。MySQL分区表作为一种有效的数据管理技术,能够将大型表划分为多个更小、更易管理的分区,从而提升数据库的性能和可...

MySQL基础篇:DQL数据查询操作_mysql 查

一、基础查询DQL基础查询语法SELECT字段列表FROM表名列表WHERE条件列表GROUPBY分组字段列表HAVING分组后条件列表ORDERBY排序字段列表LIMIT...

MySql:索引的基本使用_mysql索引的使用和原理

一、索引基础概念1.什么是索引?索引是数据库表的特殊数据结构(通常是B+树),用于...