首页 > 其他分享 >C语言 15 预处理

C语言 15 预处理

时间:2024-09-27 17:39:15浏览次数:7  
标签:15 Student int C语言 student 头文件 include 预处理 函数

C 语言学习已经快要接近尾声了,但是有一个东西迟迟还没有介绍,就是一直在写的:

#include <stdio.h>

这到底是个什么东西,为什么每次都要加上呢?这里将详细讨论它缘由。

C 语言中带 # 号的指令并不是 C 关键字的一部分,不属于 C 语言。

# 号的指令是写给编译器看的,告诉它一些事情,好让它更好的为 C 代码服务。

比如 #include</font> 指令就是告诉编译器看到这句话就要把我写的文件包含进来,#define 指令就是告诉编译器看到这个宏就用前面已经定义好的内容替换。

文件包含

当预处理器发现#include指令时,会查看后面的文件名并把文件的内容包含到当前文件中,来替换掉#include指令。比如:

int main() {
    printf("Hello World!");
}

这个函数是由系统提供的函数,实际上这个函数是在其他源文件中定义好的,而定义这个函数的源文件,就是stdio.h,可以点进去看看:

除了printf之外,看到还有很多很多的函数原型定义,他们都写到这个源文件中,而这个文件并不是以.c结尾的,而是以.h结尾的,这种文件称为头文件。头文件一般仅包含定义一类的简单信息,只要能让编译器认识就行了。

#include则是将这些头文件中提供的信息包含到 C 语言源文件中,这样才能使用定义好的printf函数,如果不添加这个指令的话,那么会:

直接不认识了,如果不告诉编译器这个函数是从哪来的,它怎么知道这个函数的具体定义是什么,程序又该怎么执行呢?

#include的具体使用格式如下:

#include <文件名称>

当然也可以写成:

#include "文件名称"

这两种写法虽然都能引入头文件,但是区别还是有的:

  • 尖括号: 引用的是编译器的库路径里面的头文件。
  • 双引号: 引用的是程序目录中相对路径中的头文件,如果找不到再去上面的库里面找。

可以看到系统已经提供好了多种多样的头文件了,通过这些系统提供的库,就可以做很多的事情了。

当然也可以自己编写一个头文件,直接在项目根目录下创建一个新的 C/C++ 头文件 test.h

// 声明函数原型
int test(int a, int b);

接着就可以编写源文件 test.c 引入这个头文件了:

#include <stdio.h>
// 因为是自己项目目录中的,所以需要使用双引号
#include "test.h" 

int main() {
    // 这样就可以使用头文件中声明的函数了
    int c = test(1, 2);
    printf("%d", c);
}

// 编写函数具体实现
int test(int a, int b) {   
    return a + b;
}
3

实际上预处理器正是通过头文件得到编译代码时所需的一些信息,然后才能把程序需要的内容(比如这里要用到的 test函数)替换到源文件中,最后才能正确编译为可执行程序。

比如现在要做一个学生管理库 student.h,这个库中提供了学生结构体的定义,以及对学生信息相关操作:

// 学生结构体定义
struct stu {
    int id;
    int age;
    char name[20];
} typedef Student;

// 打印学生信息
void print(Student* student);

// 修改年龄
void modifyAge(Student* student, int newAge);

// 修改学号
void modifyId(Student* student, int newId);

再定义源文件 student.c

#include <stdio.h>
#include "student.h"

int main() {
    Student student = {1, 18, "小明"};
    modifyAge(&student, 19);
    print(&student);
}

void print(Student* student) {
    printf("ID: %d, 姓名: %s, 年龄: %d岁\n", student->id, student->name, student->age);
}

void modifyAge(Student* student, int newAge) {
    student->age = newAge;
}

void modifyId(Student* student, int newId) {
    student->id = newId;
}
ID: 1, 姓名: 小明, 年龄: 19岁

通过使用#include就可以将项目拆分成多个模块去进行编写了。


环境:

  • GCC 11.4.0
  • VSCode 1.93.1

标签:15,Student,int,C语言,student,头文件,include,预处理,函数
From: https://www.cnblogs.com/skysailstar/p/18436213

相关文章

  • c语言数据类型和变量(下)
    2,变量C语⾔中把经常变化的值称为变量,不变的值称为常量。创建变量的语法形式:data_typename; ||||数据类型变量名变量在创建的时候就给⼀个初始值,就叫初始化。intage=18;charch='w';doubleweight=48.0;unsignedintheight=100;2.2变量的分类•全......
  • c语言数据类型和变量(上)
    C语⾔数据类型和变量数据类型介绍常见来说,使⽤整型类型来描述整数,使⽤字符类型来描述字符,使⽤浮点型类型来描述⼩数。C语言的数据类型分为内置类型和自定义类型,这里主要介绍内置类型。1.1字符型(char)char//character[signed]char//有符号的unsignedchar//⽆符号的1......
  • 实验1 C语言开发环境使用和数据类型、运算符、表达式
    任务1:1#include<stdio.h>23intmain(){4printf("OO\n");5printf("<H><H>\n");6printf("IIII\n");7return0;8} 1#include<stdio.h>23intmain(......
  • Leetcode 154. 寻找旋转排序数组中的最小值 II
    1.题目基本信息1.1.题目描述已知一个长度为n的数组,预先按照升序排列,经由1到n次旋转后,得到输入数组。例如,原数组nums=[0,1,4,4,5,6,7]在变化后可能得到:若旋转4次,则可以得到[4,5,6,7,0,1,4]若旋转7次,则可以得到[0,1,4,4,5,6,7]注意,数组[a[0],a[1],a[2],......
  • [ARC115E] LEQ and NEQ 题解
    我这场打的VP,结果E思考的时间比A还少。。但是我觉得我能想出这道题还是很有意义的,写篇题解记录一下。首先应该都不难想到动态规划吧?我们先使用暴力DP:设\(dp_{i,j}\)表示处理完前\(i\)个数,第\(i\)个数为\(j\)的方案数。我们考虑进行分类讨论:\(a_i≥a_{i-1}\):此时......
  • 实验1 C语言输入输出和简单程序编写
    实验1:源代码1:#include<stdio.h>#include<stdlib.h>intmain(){printf("O\v");printf("<H>\v");printf("II\v");printf("O\v");printf("<H>\v");printf(......
  • 【重生之我】初学C语言之 指针
    指针or地址???指针概念C语言中每个内存单元都有一个编号,有了这个内存单元的编号,CPU就可以快速找到一个内存空间,储存空间中一个字节里面能存放8个比特位。bitBYTEKBMBGBTB比特字节(8bit)1024字节1024KB1024MB1024GB内存单元的编号==地址==指针Cpu从内存单元中拿数据,Cpu算完后数据再储存在......
  • [CERC2015] Digit Division 题解
    \(O(n^2)\)做法和大部分人最开始一样,我也想的是DP。设\(dp_i\)表示用前面\(i\)个字符拆分得到的答案。既然是统计方案数,我们肯定是根据前面的答案累加。考虑在\([1,i-1]\)中选择一个\(j\),如果\([j+1,i]\)的字符组成的数字能够被\(m\)整除,那么\(dp_i\)就可以累加......
  • [JLOI2015] 有意义的字符串 题解
    拿到题目,我们首先分析一下这个奇怪的式子:\[\lfloor(\frac{b+\sqrt{d}}{2})^n\rfloor~\text{mod}~p\]重点肯定是在里面的那个式子里面,最显眼的肯定也就是那个\(\sqrt{d}\),根据整体形式,我们可以联系一元二次方程的求根公式\(x=-\dfrac{-b\pm\sqrt{b^2-4ac}}{2a}\),这里也是一......
  • 面试经典 150 题:力扣88. 合并两个有序数组
    每周一道算法题启动题目【题目链接】【解法一】合并后排序排序后的数组自动省略0的数字,又学到了classSolution{public:voidmerge(vector<int>&nums1,intm,vector<int>&nums2,intn){//合并两个数组后排序for(inti=0;i<n;i++)......