C语言:预处理

news2024/12/28 1:54:35

C语言:预处理

    • 预定义符号
    • #define
      • 定义常量
      • 定义宏
      • 宏与函数对比
    • #操作符
    • ##操作符
    • 条件编译
    • 头文件包含
      • 库文件包含
      • 本地文件包含
      • 嵌套文件包含


预定义符号

C语⾔设置了⼀些预定义符号,可以直接使⽤,预定义符号也是在预处理期间处理的。

__FILE__ //进⾏编译的源⽂件
__LINE__ //⽂件当前的⾏号
__DATE__ //⽂件被编译的⽇期
__TIME__ //⽂件被编译的时间
__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义

示例:

printf("file:%s line:%d\n", __FILE__, __LINE__);

输出结果:

file:test.c line:1

以上预定义符号,都是一些常量,可以自己一一尝试。


#define

在C语言中,#define是一个预处理器指令,用于定义宏。

宏是一个被预处理器替换的标识符或一个标识符和参数的组合。宏定义可以用来简化代码、提高代码的可读性和可维护性。

使用#define可以定义常量、函数宏等。

定义常量

  • 定义常量:
 #define PI 3.14

这样就可以在代码中使用PI来代表3.14

  • 定义关键字:
#define reg register

reg 定义为关键字 register,可以将reg这个简写代替register关键字使用。

  • 定义代码段
#define CASE break;case

正常的switch语句每一个case都要加上break,通过这个写法,我们可以在写CASE时自动补齐break


定义宏

#define 机制允许把参数替换到文本中,这种功能叫做 宏(macro) / 定义宏(define macro)

语法:

#define name(parament-list) stuff

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

示例:

#define SQUARE(x) x * x

当我们在代码中输入以下代码:

int main()
{
	int a = 5;
	int b = SQUARE(a);
	return 0;
}

在编译后就会转化为:

int main()
{
	int a = 5;
	int b = a * a;
	return 0;
}

也就是直接发生了文本替换,这种宏的形式非常像函数,因此也可以称为宏函数。但是其也有很多需要注意的地方。

比如以下代码:

int a = 5;
int b = SQUARE(a + 1);

我们希望先执行a + 1,然后再传入SQUARE中,但是其不会这样做因为其会将上述代码直接替换为:

int a = 5;
int b = a + 1 * a + 1;

由于操作符优先级的问题,我们不会得到想要的结果。为了处理这个情况,我们需要把参数用小括号括起来:

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

代码就变成:

int a = 5;
int b = (a + 1) * (a + 1);

这样我们就可以行使预期的功能了。

那么我们再看到一串代码:

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

int a = 5;
int b = 10 * DOUBLE(a);

代码编译后为:

int a = 5;
int b = 10 * 5 + 5;  

又出现了一样的问题,我们的 10 * DOUBLE(a)并没有先执行DOUBLE(a),而展开后,又出现了操作符优先级问题,所以我们的宏还要再优化:

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

在宏的最外侧再加一层括号,就可以独立运行,不受外界操作符影响了。

通过以上推断,我们可以发现,宏虽然可以很好的替换代码,但是会受到外界操作符的影响,此时就要注意很多细节。


宏与函数对比

在这里插入图片描述
函数在调用时,是会开辟内存创建栈帧的,而宏则直接执行,所以速度更快。但是由于宏是在编译阶段就已经处理好了,所以宏不能通过调试观察现象,还要操作符优先级带来的种种问题。因此宏不适合处理复杂的函数,但是很适合短小简单的函数


#操作符

功能:

#可以将宏的参数转化为字符串

比如以下代码:

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

我们尝试调用:

int a = 5;
PRINT(a);

代码就会被转化为:

int a = 5;
printf("the value of ""a" " is %d", a);

可以看到,两个n的替换效果是不同的,对于n其会直接被替换为变量a;而对于#n,其不是简单的替换,而是把参数名转化为了字符串”a“


##操作符

##操纵符可以将两个符号合并为一个符号

示例:

#define GENERIC_MAX(type) \
type type##_max(type x, type y)\
{ \
return (x>y?x:y); \
}

该宏用于创建不同类型的比大小函数,由于不同函数需要不同的函数名,于是利用##来连接函数名,也就是type##_max部分。type是一个宏参数,当传入floattype##_max整体就被连接为float_max,当传入int,整体就被连接为int_max。也就是##起到一个连接作用。


条件编译

条件编译是C语言中的一种编译指令,用于在编译过程中根据指定的条件选择性地包含或排除某些代码。它主要是为了满足不同平台、不同编译选项或不同场景下的需求。

条件编译使用预处理指令实现,预处理指令以#开头。下面是一些常用的条件编译指令及其用法:

  1. #if / #elif / #else / #endif
    #if用于基于预处理器常量的值进行条件判断。
    #elif用于在多个条件之间进行选择。
    #else用于在没有匹配的#if#elif时执行。
    #endif用于结束条件编译块。
    示例:
   #define NUM 5
   
   #if NUM > 10
       printf("NUM is greater than 10\n");
   #elif NUM > 0
       printf("NUM is greater than 0\n");
   #else
       printf("NUM is less than or equal to 0\n");
   #endif

这个代码和C语言的if代码很像,不过多讲解了。

  1. #ifdef / #ifndef
    #ifdef用于检查一个标识符是否已经定义,如果已定义则编译后面的代码,否则跳过。
    #ifndef#ifdef相反,用于检查一个标识符是否未定义。
    示例:
#ifdef DEBUG
    printf("Debug mode enabled\n");
#endif

以上代码中,只要我们定义了DEBUG这个变量,就会输出"Debug mode enabled\n"语句。

  1. #define
    #define用于定义宏。宏是一种将一组指令作为一个整体进行替换的方式。
    示例:

    #define MAX(a, b) ((a) > (b) ? (a) : (b))
    
    int x = 10;
    int y = 20;
    int max = MAX(x, y);
    
  2. #include
    #include用于将指定的头文件包含到当前文件中。
    示例:

    #include <stdio.h>
    
    int main() {
        printf("Hello, World!\n");
        return 0;
    }
    
  3. #pragma
    #pragma用于向编译器发出特定的指令,如优化选项、警告控制等。它的语法和功能因编译器而异。
    示例:

    #pragma warning(disable: 4996)
    

这些是C语言中常用的条件编译指令和代码用法。通过合理使用条件编译,我们可以根据不同的需求自由地控制代码的编译过程。


头文件包含

头文件包含分两种形式:本地头文件库文件

库文件包含

语法:

#include <filename.h>

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

我们平常使用的库文件都通过尖括号<>来包含,其会直接到存放库文件的路径中查找。

本地文件包含

#include "filename"

先在源⽂件所在⽬录下查找,如果该头⽂件未找到,编译器就像查找库函数头⽂件⼀样在标准位置查找头⽂件,如果找不到就提⽰编译错误

如果是用户自己编写的头文件,我们要用双引号""包含,如果通过这种方式包含头文件,那么会先在当前源文件的目录下查找,如果没有找到,再去库文件中查找

也就是说:库文件也可以通过双引号包含,但是会多出额外的查找步骤,所以库文件还是用尖括号包含更好,而自己编写的头文件必须双引号包含。

嵌套文件包含

假设我们现在有以下文件结构:
头文件test.h

void test();
struct Stu
{
	int id;
	char name[20];
};

源文件test.c

#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
#include "test.h"
int main()
{
	return 0;
}

我们在test.c中多次包含了头文件,这就会导致test.h反复被展开,产生大量重复代码。这就是嵌套文件包含的问题,那么我们要如何处理这个问题,让其只能被包含一次呢?

条件编译方法
在头文件test.h中加入以下代码:

#ifndef __TEST_H__
#define __TEST_H__
//头⽂件的内容
#endif //__TEST_H__

第一次包含头文件:
先执行#ifndef __TEST_H__,我们此时没有定义__TEST_H__这个变量,if成立,此时头文件会被展开,同时执行#define __TEST_H__,此时__TEST_H__就已经被定义了


第二次包含头文件:
第二次站时,由于上一次展开已经定义了__TEST_H__这个变量,导致#ifndef __TEST_H__判断为假,此时整个头文件都不会再被编译,直接舍弃

后续再展开头文件,都会因为 __TEST_H__被定义而不会编译,解决了嵌套编译的问题。

一般而言,我们这个用于判断头文件有没有被展开过的变量,是头文件名通过一定规则转化来的:

  1. 在头文件前后加上两个下划线__头文件.h__
  2. 把头文件中的点.也改为下划线__头文件_h__

因此test.h的常量就是:__TEST_H__

pragma
通过条件编译其实是比较传统的写法,我们还有一种更加简洁方便的写法:

#pragma once

只要在任何头文件前面加上这句话,就只会被编译一次了。


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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

惠普GT5810打印机报错E9的处理方法

当打印机检测到供墨系统需要维护时&#xff0c;将会出现 E9 错误。 吴中函 打印出的带错误的供墨系统维护页包含解决该错误的说明。 出现 E9 警告时维持 HP Ink Tank 打印机的打印质量&#xff0c;出现 E9 警告时如何维持 HP Ink Tank 打印机的打印质量。 惠普5810报错E9通常…

UE 打包窗口及鼠标状态设置

UE 打包窗口及鼠标状态设置 打包后鼠标不锁定 显示鼠标图标 打包后设置窗口模式 找到打包路径下的配置文件GameUserSettings&#xff0c;设置相关项目 FullscreenMode0表示全屏模式&#xff0c;1表示窗口全屏模式&#xff0c;2表示窗口模式

Spring的Bean的生命周期 | 有图有案例

Spring的Bean的生命周期 Spring的Bean的生命周期整体过程实例化初始化服务销毁循环依赖问题 完整生命周期演示 Spring的Bean的生命周期 Spring Bean的生命周期&#xff1a;从Bean的实例化之后&#xff0c;通过反射创建出对象之后&#xff0c;到Bean称为一个完整的对象&#xf…

位运算---求n的二进制表示中第k位是1还是0 (lowbit)

操作&#xff1a; 先把第k位移到最后一位&#xff08;右边第一位&#xff09; 看个位是1还是0 lowbit(x)&#xff1a;返回x的最右边的1。 原理&#xff1a; 其中 &#xff0c;意思是 是 的补码。 就可以求出最右边的一位1。 应用&#xff1a; 当中 的个数。 int re…

案例介绍:汽车维修系统的信息抽取技术与数据治理应用(开源)

一、引言 在当今汽车产业的快速发展中&#xff0c;软件已经成为提升车辆性能、安全性和用户体验的关键因素。从车载操作系统到智能驾驶辅助系统&#xff0c;软件技术的进步正在重塑我们对汽车的传统认知。我有幸参与了一个创新项目&#xff0c;该项目专注于开发和集成先进的汽…

分布式ID生成算法|雪花算法 Snowflake | Go实现

写在前面 在分布式领域中&#xff0c;不可避免的需要生成一个全局唯一ID。而在近几年的发展中有许多分布式ID生成算法&#xff0c;比较经典的就是 Twitter 的雪花算法(Snowflake Algorithm)。当然国内也有美团的基于snowflake改进的Leaf算法。那么今天我们就来介绍一下雪花算法…

2024年智能驾驶年度策略:自动驾驶开始由创造型行业转向工程型行业

感知模块技术路径已趋于收敛&#xff0c;自动驾驶从创造型行业迈向工程型行业。在特斯拉的引领下&#xff0c;国内主机厂2022年以来纷纷跟随特斯拉相继提出“重感知、轻地图”技术方案&#xff0c;全球自动驾驶行业感知模块技术路径从百花齐放开始走向收敛。我们认为主机厂智能…

光学遥感卫星分辨率的奥秘 !!

文章目录 前言 1、光学遥感卫星分辨率的多维视角 &#xff08;1&#xff09;空间分辨率 &#xff08;2&#xff09;光谱分辨率 &#xff08;3&#xff09;辐射分辨率 &#xff08;4&#xff09;时间分辨率 2、光学遥感分辨率的重要性 3、遥感分辨率的挑战与进步 4、未来展望 总…

Git推送本地仓库至阿里云仓库

Git推送本地仓库至阿里云仓库 1.安装Git 参考Git安装详解 2.生成 SSH 密钥 基于RSA算法SSH 密钥 1.管理员权限运行Git Bash 2.输入生成密钥指令点击回车&#xff0c;选择 SSH 密钥生成路径。 $ ssh-keygen -t rsa -C "2267521563qq.com"3.以 RSA算法为例&…

ABAP - SALV教程17 弹窗ALV

SALV可以通过弹窗形式打开在生成SALV实例对象后调用set_screen_popup方法设置成弹出模式 "设置为弹窗模式 go_alv->set_screen_popup( start_column 10end_column 110start_line 5end_line 15). 显示效果 完整代码 SELECT *FROM ekkoINTO TABLE DATA(gt_dat…

C++自学精简实践教程

一、介绍 1.1 教程特点 一篇文章从入门到就业有图有真相&#xff0c;有测试用例&#xff0c;有作业&#xff1b;提供框架代码&#xff0c;作业只需要代码填空规范开发习惯&#xff0c;培养设计能力 1.2 参考书 唯一参考书《C Primer 第5版》​参考书下载&#xff1a; 蓝奏云…

如何自己系统的学python

学习Python是一项很好的投资&#xff0c;因为它是一种既强大又易于学习的编程语言&#xff0c;适用于多种应用&#xff0c;如数据分析、人工智能、网站开发等。下面是一个系统学习Python的步骤建议&#xff1a; 基础准备 安装Python&#xff1a; 访问Python官网下载最新版本的…

代码随想录第45天|● 198.打家劫舍 ● 213.打家劫舍II ● 337.打家劫舍III

文章目录 ● 198.打家劫舍思路代码1.dp数组两个变量 ● 213.打家劫舍II思路&#xff1a;代码 ● 337.打家劫舍III思路代码&#xff1a; ● 198.打家劫舍 思路 代码 1.dp数组 class Solution {public int rob(int[] nums) {if(nums.length1)return nums[0];int[] dpnew int[nu…

CentOS下安装Kafka3

kafka是分布式消息队列&#xff0c;本文讲述其在centos&#xff08;centos 7.5&#xff09;下的安装。安装过程可以参考其官方文档https://kafka.apache.org/36/documentation.html 首先在官网 https://kafka.apache.org/downloads 下载Kafka二进制文件&#xff08;官网的压缩包…

WordPress建站入门教程:如何在本地电脑搭建WordPress网站?

前面跟大家分享了『WordPress建站入门教程&#xff1a;如何安装本地WordPress网站运行环境&#xff1f;』&#xff0c;接下来boke112百科就继续跟大家分享本地电脑如何搭建WordPress网站。 小皮面板&#xff08;phpstudy&#xff09;的“软件管理 – 网站程序”虽然可以一键部…

2023年,我的年终总结

序言 2023年的年终总结一直拖到现在&#xff0c;想来是有多个原因吧&#xff1a;第一个应该是年底还有些事情没有完成&#xff0c;内心有所不甘&#xff1b;第二个应该是这一年似乎是很忙碌的一年&#xff0c;不知从何说起&#xff1b;第三个应该是对于自己这一年的收获&#…

AD22编译他人分享的集成库

Content 1. 新建集成库2. 添加工程文件3. 编译工程 从网上下载了很多封装库&#xff0c;但是不知道如何添加进去&#xff0c;折腾了一会儿弄明白了 1. 新建集成库 2. 添加工程文件 添加完之后是这个样子&#xff0c;总而言之就是把你需要的全部添加进来&#xff1a; 3. 编译工…

ER-NeRF实时对话数字人模型训练与部署

ER-NeRF是基于NeRF用于生成数字人的方法&#xff0c;可以达到实时生成的效果。 下载源码 cd D:\Projects\ git clone https://github.com/Fictionarry/ER-NeRF cd D:\Projects\ER-NeRF 下载模型 准备面部解析模型 wget https://github.com/YudongGuo/AD-NeRF/blob/master/…

STM32(6)中断

1.中断 1.1 中断的概念 STM32的中断&#xff1a; 1.2 中断优先级 用数字的大小表示中断优先级的高低&#xff0c;数字的范围&#xff1a;0000--1111&#xff08;二进制&#xff09;&#xff0c;即0-15&#xff0c;共16级优先级。 进一步对这4位二进制数进行划分&#xff0c;可…

java012 - Java集合基础

1、集合基础 1.1 集合概述 引用数据类型包括&#xff1a;类、接口、数组[] 1.2 ArrayList构造和添加方法 代码&#xff1a; 空集合对象&#xff1a;[] add() add(int index,E element): 1.3 ArrayList集合常用方法