文件包含的本质、预处理符号、# vs ##
何为头文件?
在C语言中,文件包含是一种常见的编程技术,它允许程序员在一个源文件中使用另一个源文件中的函数或变量。
文件包含通常使用`#include`预处理指令来实现。`#include`指令告诉预处理器将文件的内容插入到当前文件的指定位置中。
例如,在一个C源文件中,如果想要使用另一个源文件中的函数,可以使用以下语句:
#include "otherfile.c"这个语句会告诉编译器将`otherfile.c`中的代码插入到当前文件的位置,然后再进行编译。当编译器遇到调用`otherfile.c`中的函数时,它能够找到函数的定义,并将它们编译到可执行文件中。
需要注意的是,文件包含应该遵循一些最佳实践:
- 为了避免重复包含,应该使用头文件而不是源文件进行文件包含。例如,使用`#include "otherfile.h"`而不是`#include "otherfile.c"`。
#include "otherfile.h" //使用 #include "otherfile.c" //不使用- 应该避免在头文件中放置函数或变量的定义。头文件应该只包含函数和变量的声明。
- 应该避免在头文件中使用全局变量。全局变量会在包含文件的每个源文件中创建一个独立的实例,这样可能会导致命名冲突和意外行为。
为何所有头文件,都推荐写入下面代码?本质是为什么?
#ifndef XXX
#define XXX//TODO#endif这是为了避免头文件重复包含多次,导致编译错误或者不必要的浪费。当一个头文件被多次包含时,如果没有预处理器指令的保护,就会重复定义同一个符号,从而出现编译错误。
为了避免这种问题,使用了 `#ifndef`、`#define`、`#endif` 三个预处理器指令,将头文件的内容包含在一个条件编译的块中。第一次包含头文件时,`XXX`未被定义,`#ifndef` 判断为真,进入条件编译块,`#define XXX` 定义符号 `XXX`,然后包含头文件的内容。
当再次包含同一头文件时,`XXX`已被定义,`#ifndef` 判断为假,直接跳过条件编译块,从而避免了重复定义的问题。
#include究竟干了什么?
#include本质是把头文件中相关内容,直接拷贝至源文件中!
那么,在多文件包含中,有没有可能存在头文件被重复包含,乃至被重复拷贝的问题呢?
test.h
#ifndef _TEST_H_
#define _TEST_H_ //注意,这里没有包含<stdio.h>防止信息太多干扰我们
extern void show(); //任意一个函数声明
#endif
test.c
#include "test.h" //故意包含两次
#include "test.h"
int main()
{return 0;
}经过预编译后的结果
# 1 "test.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "test.c"
# 1 "test.h" 1 //test.h只被包含了1次extern void show();
# 2 "test.c" 2
int main()
{return 0;
}但是当我们去掉条件编译呢?
# 1 "test.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "test.c"
# 1 "test.h" 1extern void show(); //内容被拷贝第一次
# 2 "test.c" 2
# 1 "test.h" 1extern void show(); //内容被拷贝第二次
# 3 "test.c" 2
int main()
{return 0;
}
结论:
所有头文件都必须带上条件编译,防止被重复包含! 那么,重复包含一定报错吗?不会! 重复包含,会引起多次拷贝,主要会影响编译效率!同时,也可能引起一些未定义错误,但是特别少。
#error 预处理
`#error`指令是C语言预处理器中的一个预编译指令,用于在预处理阶段产生编译错误。
#include <stdio.h>
#define __welcome
int main()
{
#ifdef __welcome
#error 老铁,非常感谢观看此篇文章哟!!!
#endifreturn 0;
}
程序使用了条件编译指令`#ifdef`和`#endif`来判断`__welcome`宏是否已经定义。如果已经定义,则使用`#error`指令生成一个编译错误,输出一条提示信息。
#include <stdio.h>
//#define __welcome
int main()
{
#ifndef __welcome
#error 老铁,非常感谢观看此篇文章哟!!!
#endifreturn 0;
}
程序定义了一个名为`__welcome`的宏,并使用了条件编译指令`#ifndef`和`#endif`来判断`__welcome`宏是否已经定义。如果未定义,则使用`#error`指令生成一个编译错误,输出一条提示信息。
结论:核心作用是可以进行自定义编译报错。
#line 预处理
`#line`指令是C语言预处理器中的一个预编译指令,用于更改源代码中的行号和文件名,从而影响编译器错误和警告信息中的行号和文件名。
//本质其实是可以定制化你的文件名称和代码行号,很少使用
#include <stdio.h>
int main()
{printf("%s, %d\n", __FILE__, __LINE__); //C预定义符号,代表当前文件名和代码行号
#line 60 "welcome.h" //定制化完成printf("%s, %d\n", __FILE__, __LINE__);return 0;
}程序使用了`#line`指令将当前行号设置为60,并将当前文件名设置为`"welcome.h"`,从而定制化了文件名称和代码行号。在后续的`printf`函数中,预处理器会将`__FILE__`和`__LINE__`再次替换成定制化后的文件名和行号,从而输出新的信息。

#pragma 预处理
`#pragma`指令是一种不可依赖的、非标准的预处理指令,它通常被编译器用来提供一些与平台、编译器或者其他特殊需求相关的功能。
一些常见的`#pragma`指令包括:
- `#pragma once`:告诉编译器只包含一次某个头文件,避免重复定义。
- `#pragma GCC optimize`:指示GCC编译器优化代码。
- `#pragma warning`:指示编译器输出警告信息。
- `#pragma pack`:指示编译器对结构体进行字节对齐。
- `#pragma message()`:可以用来进行对代码中特定的符号(比如其他宏定义)进行是否存在进行编译时消息提醒。
#include <stdio.h>
#define READ 
int main()
{
#ifdef READ
#pragma message("谢谢宝子阅读文章!!!")
#endifreturn 0;
}
# 运算符
`#`运算符是C/C++语言中的一个预处理运算符,用于将宏定义参数转换成字符串常量。
#include<stdio.h>
int main()
{printf("hello world\n");printf("hello""world""\n");const char* msg = "hello""world""\n";//printf("%s\n",msg);printf(msg);return 0;
}
结论:相邻字符串自动连接特性
#include<stdio.h>
#define STR(s) #s
int main()
{printf("PI = "STR(3.1415926)"\n");return 0;
}

## 预算符
`##`预算符是C/C++语言中的一个预处理运算符,用于将两个符号拼接成一个新的符号。
#include<stdio.h>
#define XNAME(n) student##n
int main()
{XNAME(1);XNAME(2);XNAME(3);XNAME(4);XNAME(5);XNAME(6);return 0;
}
##的实质:将##相连的两个符号,连接成为一个符号。
小练习实例:计算一个的科学计数法值
#include<stdio.h>#define CONT(x,n) (x##e##n)
int main()
{//计算浮点数科学计数法,相当于1.1 * (10^2)printf("%f\n", 1.1e2);printf("%f\n", CONT(1.1, 2)); return 0;
}
这段代码定义了一个宏`CONT`,用于将两个参数拼接成一个科学计数法格式的浮点数。在`main`函数中,首先使用`1.1e2`的科学计数法直接输出了`110.000000`,然后使用`CONT(1.1, 2)`宏对参数进行拼接,得到了相同的结果。由于`1.1`和`2`经过了拼接,因此最终展开的结果相当于`1.1e2`,即`110.000000`。这说明了`##`预算符可以用于将数字、字符串、变量名等不同类型的记号拼接在一起,从而得到想要的结果。
#include <stdio.h>#define CONCAT(x, y) x##yint main() {int xy = 10;printf("%d\n", CONCAT(x, y));//10return 0;
}
使用`##`运算符和`CONCAT`宏定义输出变量`xy`的值。首先,在`main`函数中定义了一个名为`xy`的整型变量,并赋值为`10`。然后,通过`CONCAT(x, y)`宏调用,将参数`x`和`y`拼接在一起,得到了记号`xy`。最后,使用`printf`函数输出了`xy`变量的值,结果为`10`。由此可见,`##`运算符可以将字符串、变量名等记号拼接在一起,从而实现更加灵活的程序设计。
相关文章:
 
文件包含的本质、预处理符号、# vs ##
何为头文件? 在C语言中,文件包含是一种常见的编程技术,它允许程序员在一个源文件中使用另一个源文件中的函数或变量。 文件包含通常使用#include预处理指令来实现。#include指令告诉预处理器将文件的内容插入到当前文件的指定位置中。 例如&a…...
 
【JavaSE】Java基础语法(三十九):网络编程入门
文章目录 1. 网络编程概述2. 网络编程三要素3. IP地址4. InetAddress5. 端口和协议 1. 网络编程概述 计算机网络 是指将地理位置不同的具有独立功能的多台计算机及其外部设备,通过通信线路连接起来,在网络 操作系统,网络管理软件及网络通信协…...
 
中间件SOME/IP简述
SOME/IP SOME/IP 不是广义上的中间件,严格的来讲它是一种通信协议,但中间件这个概念太模糊了,所以我们也一般称 SOME/IP 为通信中间件。 SOME/IP 全称是 Scalable service-Oriented MiddlewarE over IP。也就是基于 IP 协议的面向服务的可扩…...
 
[自学记录03|百人计划]移动端GPU的TB(D)R架构基础
一、专有名词解释 1.System on Chip(Soc) Soc是把CPU、GPU、内存、通信基带、GPS模块等等整合在一起的芯片的称呼。常见有A系Soc(苹果),骁龙Soc(高通),麒麟Soc(华为&am…...
 
详解Java枚举
一、知识点 二、概念 enum 的全称为 enumeration, 是 JDK 1.5 中引入的新特性。 在Java中,被 enum 关键字修饰的类型就是枚举类型。形式如下: enum Color { RED, GREEN, BLUE }如果枚举不添加任何方法,枚举值默认为从0开始的有…...
 
ES6-ES13学习笔记(4.0)
includes函数 判断字符串是否存在指定字符 <!--* Author: RealRoad1083425287qq.com* Date: 2023-06-01 08:40:33* LastEditors: Mei* LastEditTime: 2023-06-01 08:58:54* FilePath: \vscode\ECMA\05\01.html* Description: * * Copyright (c) 2023 by ${git_name_ema…...
 
线段树C++详细讲解和个人见解
问题引入 1275. 最大数 给定一个正整数数列 a1,a2,…,an,每一个数都在 0∼p−1 之间。 可以对这列数进行两种操作: 添加操作:向序列后添加一个数,序列长度变成 n1;询问操作:询问这个序列中最后 L 个数中…...
构建sysbench的镜像
方式1:先docker run一个镜像,手动安装好commit docker run -it --name mycentos arm64v8/centos:7 /bin/bash docker commit -a "PX Bai" mycentos mycentos1 docker run -it -d --namemycentos1 mycentos1 /bin/bash docker exec -it mycent…...
leetcode解题思路分析(一百四十)1201 - 1208 题
丑数3 给你四个整数:n 、a 、b 、c ,请你设计一个算法来找出第 n 个丑数。丑数是可以被 a 或 b 或 c 整除的 正整数 。 容斥原理二分法 class Solution { public:int nthUglyNumber(int n, int a, int b, int c) {long long ab lcm((long long)a, (lo…...
FPGA设计的指导性原则 (一)
这一部分主要介绍FPGA/CPLD设计的指导性原则,如FPGA设计的基本原则、基本设 计思想、基本操作技巧、常用模块等。FPGA/CPLD设计的基本原则、思想、技巧和常用模 块是一个非常大的问题,在此不可能面面俱到,只能我们公司项目中常用的一些设计原则与 方法提纲携领地加以介绍,希…...
 
【架构】常见技术点--服务治理
导读:收集常见架构技术点,作为项目经理了解这些知识点以及解决具体场景是很有必要的。技术要服务业务,技术跟业务具体结合才能发挥技术的价值。 目录 1. 微服务 2. 服务发现 3. 流量削峰 4. 版本兼容 5. 过载保护 6. 服务熔断 7. 服务…...
 
手撕数据结构—单链表
✅作者:简单^不简单 🔥系列专栏:C语言数据结构 💖如果文章有错误,时刻欢迎大家的指正。当然觉得博主的文章还不错的话,请点赞👍收藏⭐️留言📝 💬格言:希望我…...
 
Benewake(北醒) 快速实现 TF02-i-RS485 与电脑通信操作说明
目录 一、前言二、工具准备1. USB-RS485 转接器2. TF02-i-RS4853. 兆信直流电源4.连接线、绝缘胶带、螺丝刀5. PC:Windows 系统6. 串口助手软件 三、连接方式1. USB-RS485 转接板接口说明2. TF02-i-RS485 引脚定义3. 连接图 四、TF02-i-RS485 与电脑通信操作说明1. …...
 
【分享】科大讯飞星火认知大模型(初体验)
前言: 哈喽,大家好,我是木易巷~ 随着人工智能技术的迅猛发展,自然语言处理(NLP)成为了热门话题。在众多NLP模型中,科大讯飞星火认知大模型成为了一个备受瞩目的新秀,今天我们来了解…...
logstash 采集应用日志切割问题
1.logstash [oswatch@rce1 conf]$ cat test.conf input { file { path=>["/tmp/test/test.log*"] } } output { stdout { codec=>rubydebug{} } } 2.python脚本: [oswatch@rce1 conf]$ cat t1.py #!/usr/bin/python # -*- coding: UTF-…...
 
计算机网络实验:认识Packet Tracer软件
目录 前言实验目的实验内容及要求相关知识点实验指导实验过程总结 前言 计算机网络是当今信息技术的重要组成部分,它涉及到多种硬件和软件的协同工作,以实现数据的传输和交换。为了更好地理解和掌握计算机网络的基本原理和技术,我们需要进行…...
 
【MySQL新手到通关】第六章 时间日期函数
文章目录 1.获取日期时间函数1.1 获取当前日期时间1.2 获取当前日期1.3 获取当前时间 2.日期格式化★★★2.1 日期转指定格式字符串2.2 字符串转日期 3.日期间隔3.1 增加日期间隔 ★★★3.2 减去一个时间间隔★★★3.3 日期相差天数(天)3.4 相差时间&…...
 
深蓝学院C++基础笔记 第 1 章 C++初探
第 1 章 C初探 1.从Hello World 谈起 Hello World: #include <iostream> int mian() { std::cout << "Hello World!" << std::endl; }函数: 一段能被反复调用的代码,可以接收输入,进行处理并(或)产生输出-返回…...
 
【配电网重构】基于混合整数二阶锥配电网重构研究(Matlab代码实现)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...
Kubernetes mysql 实战以及外部存储处理 [一]
在 Kubernetes 中部署 MySQL 数据库需要考虑以下几个方面: 部署方式:可以选择使用 StatefulSet 或者 Deployment 进行部署,如果需要有状态的服务,使用 StatefulSet 更加合适。存储:MySQL 需要一个持久化存储来保存数据。可以使用 Kubernetes 提供的 PersistentVolumeClaim…...
 
【Axure高保真原型】引导弹窗
今天和大家中分享引导弹窗的原型模板,载入页面后,会显示引导弹窗,适用于引导用户使用页面,点击完成后,会显示下一个引导弹窗,直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…...
 
Prompt Tuning、P-Tuning、Prefix Tuning的区别
一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...
Oracle查询表空间大小
1 查询数据库中所有的表空间以及表空间所占空间的大小 SELECTtablespace_name,sum( bytes ) / 1024 / 1024 FROMdba_data_files GROUP BYtablespace_name; 2 Oracle查询表空间大小及每个表所占空间的大小 SELECTtablespace_name,file_id,file_name,round( bytes / ( 1024 …...
【Java学习笔记】Arrays类
Arrays 类 1. 导入包:import java.util.Arrays 2. 常用方法一览表 方法描述Arrays.toString()返回数组的字符串形式Arrays.sort()排序(自然排序和定制排序)Arrays.binarySearch()通过二分搜索法进行查找(前提:数组是…...
 
DAY 47
三、通道注意力 3.1 通道注意力的定义 # 新增:通道注意力模块(SE模块) class ChannelAttention(nn.Module):"""通道注意力模块(Squeeze-and-Excitation)"""def __init__(self, in_channels, reduction_rat…...
 
深入理解JavaScript设计模式之单例模式
目录 什么是单例模式为什么需要单例模式常见应用场景包括 单例模式实现透明单例模式实现不透明单例模式用代理实现单例模式javaScript中的单例模式使用命名空间使用闭包封装私有变量 惰性单例通用的惰性单例 结语 什么是单例模式 单例模式(Singleton Pattern&#…...
将对透视变换后的图像使用Otsu进行阈值化,来分离黑色和白色像素。这句话中的Otsu是什么意思?
Otsu 是一种自动阈值化方法,用于将图像分割为前景和背景。它通过最小化图像的类内方差或等价地最大化类间方差来选择最佳阈值。这种方法特别适用于图像的二值化处理,能够自动确定一个阈值,将图像中的像素分为黑色和白色两类。 Otsu 方法的原…...
 
【Oracle】分区表
个人主页:Guiat 归属专栏:Oracle 文章目录 1. 分区表基础概述1.1 分区表的概念与优势1.2 分区类型概览1.3 分区表的工作原理 2. 范围分区 (RANGE Partitioning)2.1 基础范围分区2.1.1 按日期范围分区2.1.2 按数值范围分区 2.2 间隔分区 (INTERVAL Partit…...
 
排序算法总结(C++)
目录 一、稳定性二、排序算法选择、冒泡、插入排序归并排序随机快速排序堆排序基数排序计数排序 三、总结 一、稳定性 排序算法的稳定性是指:同样大小的样本 **(同样大小的数据)**在排序之后不会改变原始的相对次序。 稳定性对基础类型对象…...
 
【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
