何为头文件?
在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" 1
extern void show(); //内容被拷贝第一次
# 2 "test.c" 2
# 1 "test.h" 1
extern void show(); //内容被拷贝第二次
# 3 "test.c" 2
int main()
{
return 0;
}
结论:
所有头文件都必须带上条件编译,防止被重复包含! 那么,重复包含一定报错吗?不会! 重复包含,会引起多次拷贝,主要会影响编译效率!同时,也可能引起一些未定义错误,但是特别少。
#error 预处理
`#error`指令是C语言预处理器中的一个预编译指令,用于在预处理阶段产生编译错误。
#include <stdio.h>
#define __welcome
int main()
{
#ifdef __welcome
#error 老铁,非常感谢观看此篇文章哟!!!
#endif
return 0;
}
程序使用了条件编译指令`#ifdef`和`#endif`来判断`__welcome`宏是否已经定义。如果已经定义,则使用`#error`指令生成一个编译错误,输出一条提示信息。
#include <stdio.h>
//#define __welcome
int main()
{
#ifndef __welcome
#error 老铁,非常感谢观看此篇文章哟!!!
#endif
return 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("谢谢宝子阅读文章!!!")
#endif
return 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##y
int main() {
int xy = 10;
printf("%d\n", CONCAT(x, y));//10
return 0;
}
使用`##`运算符和`CONCAT`宏定义输出变量`xy`的值。首先,在`main`函数中定义了一个名为`xy`的整型变量,并赋值为`10`。然后,通过`CONCAT(x, y)`宏调用,将参数`x`和`y`拼接在一起,得到了记号`xy`。最后,使用`printf`函数输出了`xy`变量的值,结果为`10`。由此可见,`##`运算符可以将字符串、变量名等记号拼接在一起,从而实现更加灵活的程序设计。