C语言---预处理详解 2

简介: C语言---预处理详解

      2.7命名约定

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

所以给友友两个建议:

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

3.#undef

在C语言中,#undef是一个预处理指令,用于取消定义一个已经定义的宏。

宏定义是一种在编译时用特定的值或代码片段替换标识符的方式。通过#define指令可以定义一个宏,而#undef指令可以取消定义宏。

4.命令行定义

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

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

Linux环境下演示:

指令:

gcc test.c -D ARRAY_SIZE=100 -o test

5.条件编译

条件编译是一种预处理指令,它允许根据条件来选择性地编译代码。条件编译指令在编译阶段进行处理,可以根据条件的真假来决定是否编译代码。

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

条件编译使用#if#ifdef#ifndef#elif#else#endif等预处理指令来实现。下面列举条件编译指令:

1. #if CONDITION
2. // 代码块
3. #endif

#ifdef#endif:用于检查一个标识符是否已经定义,如果已定义,则编译条件之间的代码。

1. #ifdef IDENTIFIER
2. // 代码块
3. #endif

#ifndef#endif:与#ifdef相反,用于检查一个标识符是否未定义,如果未定义,则编译条件之间的代码。

1. #ifndef IDENTIFIER
2. // 代码块
3. #endif

#elif:用于指定一个新的条件,如果前面的条件为假,且当前条件为真,则编译条件之间的代码。

#if CONDITION1
    // 代码块1
#elif CONDITION2
    // 代码块2
#else
    // 代码块3
#endif

条件可以是任何可以求值为非零或零的表达式,通常使用预定义的宏来表示条件。例如,#ifdef指令通常用于检查是否定义了某个宏,如下所示:

#include <stdio.h>
#define DEBUG
int main() {
    #ifdef DEBUG
        printf("Debug mode\n");
    #else
        printf("Release mode\n");
    #endif
    return 0;
}

如果定义了宏DEBUG,则会编译输出"Debug mode";否则,会编译输出"Release mode"。

条件编译可以用于在不同的编译环境下编译不同的代码,或者根据不同的条件来选择性地包含或排除代码。它在处理平台特定代码、调试代码和配置选项等方面非常有用。

6.文件包含

       我们已经知道, #include 指令可以使另外一个文件被编译。就像它实际出现于 #include 指令的地方一样。

这种替换的方式很简单:

  •        预处理器先删除这条指令,并用包含文件的内容替换。
  •        这样一个源文件被包含10次,那就实际被编译10次。

6.1头文件被包含的方式:

本地文件包含:

#include "test.h"

查找策略:

       先在源文件所在目录下查找,如果该头文件未找到,编译器就像查找库函数头文件一样在标准位置查找头文件。如果找不到就提示编译错误。

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

/usr/include

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

你如果是默安装路径的话:C:\Program Files (x86)\Microsoft Visual Studio\2022\Community\VC\Tools\MSVC\14.30.30705\include

注意 : 按照自己的安装路径去找每个人的机器各有差异


256b99ef889c4e179727a97668bc5618.png

库文件包含:

#include <stdio.h>

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

这样是不是可以说,对于库文件也可以使用 “” 的形式包含?

答案是肯定的,可以。

但是这样做查找的效率就低些,当然这样也不容易区分是库文件还是本地文件了

6.2嵌套文件包含


comm.h和comm.c是公共模块。

test1.h和test1.c使用了公共模块。

test2.h和test2.c使用了公共模块。

test.h和test.c使用了test1模块和test2模块。

这样最终程序中就会出现两份comm.h的内容。这样就造成了文件内容的重复.


通过前面的学习我们应该都知道了,有不明白的友友可以看这篇程序环境


头文件在预处理的时候会直接在原地展开,如果一个一个头文件按800行代码计算,如果你重复包含多个的话,那么就会造成冗余,降低程序的运行效率.


那有没有什么办法可以避免呢?

1.条件编译

每个头文件的开头写:

#ifndef __TEST_H__   // 如果未定义标识符__TEST_H__
#define __TEST_H__   // 定义标识符__TEST_H__
// 头文件的内容
#endif //__TEST_H__

通过这种方式,当多个源文件需要包含同一个头文件时,只有第一次包含会生效,后续的包含会被忽略,避免了头文件的重复包含问题。


2.#pragma once


#pragma once是一种用于防止头文件多重包含的预处理指令。与传统的条件编译指令相比,#pragma once更简洁和直观。


使用#pragma once可以确保头文件只被包含一次。当编译器遇到#pragma once指令时,它会检查当前的头文件是否已经被包含过,如果是,则直接跳过该头文件的包含,否则继续包含该头文件。


要使用#pragma once,只需要在头文件的开头添加一行#pragma once即可

#pragma once
// 头文件的内容

本章内容已完:

       如果您喜欢这篇文章,可以点赞、评论和分享给您的同学,这将对我提供巨大的鼓励和支持。另外,我会在未来的更新中持续探讨与c/c++相关的内容。我会为您带来更多关于编程技术问题的深入解析、应用案例和趣味玩法等。感兴趣的话给博主点个关注,获取最新的内容消息!

目录
相关文章
|
2月前
|
编译器 C语言
C语言的预处理指令
C语言的预处理指令
|
2月前
|
编译器 Linux PHP
C语言从入门到实战——预处理详解
C语言预处理是C语言编译过程的一个阶段,它在编译之前对源代码进行一系列的处理操作,包括宏替换、文件包含、条件编译等,最终生成经过预处理的代码,然后再进行编译。
48 0
|
5月前
|
存储 自然语言处理 编译器
『C语言进阶』程序环境和预处理
『C语言进阶』程序环境和预处理
|
3天前
|
编译器 C语言
C语言:预处理
C语言:预处理
7 1
C语言:预处理
|
12天前
|
存储 Linux C语言
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)-2
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)
|
12天前
|
自然语言处理 Linux 编译器
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)-1
C语言进阶第十一节 --------程序环境和预处理(包含宏的解释)
|
18天前
|
编译器 Linux C语言
C语言:预处理详解(知识点和代码演示)
C语言:预处理详解(知识点和代码演示)
|
2月前
|
存储 编译器 程序员
【程序环境和预处理】C语言
【程序环境和预处理】C语言
|
2月前
|
存储 机器学习/深度学习 自然语言处理
【进阶C语言】编译与链接、预处理符号详解
【进阶C语言】编译与链接、预处理符号详解
28 0
|
2月前
|
存储 自然语言处理 编译器
【C语言进阶】程序环境和预处理
【C语言进阶】程序环境和预处理