深入C语言文件操作:从库函数到系统调用
引言
文件操作是编程中不可或缺的一部分,尤其在C语言中,文件操作不仅是处理数据的基本手段,也是连接程序与外部世界的重要桥梁。C语言提供了丰富的库函数来处理文件,如 fopen、fclose、fread、fwrite 等。然而,这些库函数实际上是基于操作系统提供的系统调用构建的。理解库函数和系统调用之间的关系,不仅有助于编写高效的代码,还能帮助我们更好地理解底层操作系统的机制。
本文将深入探讨C语言文件操作中的库函数和系统调用,解释它们的工作原理、区别和联系,并通过实际示例展示如何使用这些函数。

C标准库函数与系统调用概述
6.1 C标准库函数
C标准库函数是ANSI C标准中定义的一组函数,它们提供了一种跨平台的方式来处理文件操作。这些函数通常在 stdio.h 头文件中声明,并且在大多数操作系统中都有实现。常见的文件操作库函数包括:
fopen:打开文件。fclose:关闭文件。fread:从文件中读取数据。fwrite:向文件中写入数据。fgetc:从文件中读取一个字符。fputc:向文件中写入一个字符。fgets:从文件中读取一行。fputs:向文件中写入一行。fseek:移动文件指针。ftell:获取文件指针的当前位置。rewind:将文件指针重置到文件开头。
6.2 系统调用
系统调用是操作系统提供给用户程序的一组接口,用于请求操作系统执行特定的低级操作。系统调用通常在内核态执行,提供了对硬件设备的直接访问。常见的文件操作系统调用包括:
open:打开文件。close:关闭文件。read:从文件中读取数据。write:向文件中写入数据。lseek:移动文件指针。ioctl:控制设备。
库函数与系统调用的区别
6.3 工作空间不同
- 库函数:运行在用户态,通常包含在标准库中,如
glibc。 - 系统调用:运行在内核态,由操作系统内核提供。
6.4 缓冲机制不同
- 库函数:通常使用缓冲机制来提高性能。例如,
fread和fwrite会先将数据读取到内存缓冲区,然后再批量处理。 - 系统调用:不使用缓冲机制,每次调用都会直接与文件系统交互。
6.5 可移植性不同
- 库函数:具有良好的可移植性,可以在不同的操作系统上使用相同的接口。
- 系统调用:依赖于特定的操作系统,不同操作系统的系统调用接口可能不同。
6.6 性能差异
- 库函数:由于使用了缓冲机制,减少了用户态和内核态之间的切换次数,通常性能更高。
- 系统调用:每次调用都会导致用户态和内核态之间的切换,性能较低。
库函数与系统调用的联系
尽管库函数和系统调用在许多方面有所不同,但它们之间存在着密切的联系。实际上,许多库函数最终会调用系统调用来完成实际的文件操作。
6.7 fopen 与 open
fopen 函数用于打开文件,并返回一个指向 FILE 结构的指针。fopen 实际上调用了 open 系统调用。
示例代码:
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>int main() {// 使用 fopen 打开文件FILE *file = fopen("example.txt", "r");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}// 使用 open 系统调用打开文件int fd = open("example.txt", O_RDONLY);if (fd == -1) {perror("打开文件失败");return 1;}fclose(file);close(fd);return 0;
}
6.8 fread 与 read
fread 函数用于从文件中读取数据,并返回实际读取的数据项数。fread 实际上调用了 read 系统调用。
示例代码:
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>int main() {// 使用 fopen 打开文件FILE *file = fopen("example.txt", "r");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}// 使用 open 系统调用打开文件int fd = open("example.txt", O_RDONLY);if (fd == -1) {perror("打开文件失败");return 1;}char buffer[100];size_t bytes_read;// 使用 fread 读取文件bytes_read = fread(buffer, 1, sizeof(buffer), file);if (ferror(file)) {fprintf(stderr, "读取文件失败: %s\n", strerror(errno));fclose(file);close(fd);return 1;}buffer[bytes_read] = '\0';printf("fread: %s\n", buffer);// 使用 read 系统调用读取文件bytes_read = read(fd, buffer, sizeof(buffer));if (bytes_read == -1) {perror("读取文件失败");close(fd);return 1;}buffer[bytes_read] = '\0';printf("read: %s\n", buffer);fclose(file);close(fd);return 0;
}
6.9 fwrite 与 write
fwrite 函数用于向文件中写入数据,并返回实际写入的数据项数。fwrite 实际上调用了 write 系统调用。
示例代码:
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>int main() {// 使用 fopen 打开文件FILE *file = fopen("example.txt", "w");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}// 使用 open 系统调用打开文件int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);if (fd == -1) {perror("打开文件失败");return 1;}char *message = "Hello, World!\n";// 使用 fwrite 写入文件if (fwrite(message, 1, strlen(message), file) != strlen(message)) {if (ferror(file)) {fprintf(stderr, "写入文件失败: %s\n", strerror(errno));fclose(file);close(fd);return 1;}}// 使用 write 系统调用写入文件if (write(fd, message, strlen(message)) == -1) {perror("写入文件失败");close(fd);return 1;}fclose(file);close(fd);return 0;
}
6.10 fclose 与 close
fclose 函数用于关闭文件,并刷新缓冲区。fclose 实际上调用了 close 系统调用。
示例代码:
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>int main() {// 使用 fopen 打开文件FILE *file = fopen("example.txt", "w");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}// 使用 open 系统调用打开文件int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);if (fd == -1) {perror("打开文件失败");return 1;}char *message = "Hello, World!\n";// 使用 fwrite 写入文件if (fwrite(message, 1, strlen(message), file) != strlen(message)) {if (ferror(file)) {fprintf(stderr, "写入文件失败: %s\n", strerror(errno));fclose(file);close(fd);return 1;}}// 使用 write 系统调用写入文件if (write(fd, message, strlen(message)) == -1) {perror("写入文件失败");close(fd);return 1;}// 使用 fclose 关闭文件if (fclose(file) != 0) {fprintf(stderr, "关闭文件失败: %s\n", strerror(errno));close(fd);return 1;}// 使用 close 系统调用关闭文件if (close(fd) == -1) {perror("关闭文件失败");return 1;}return 0;
}
文件描述符与缓冲区
6.11 文件描述符
文件描述符(File Descriptor,FD)是一个用于引用打开文件和其他类型的I/O资源的整数。每个进程都有自己的文件描述符表,用于跟踪进程打开的所有文件和I/O资源。
- 唯一标识:文件描述符为每个打开的文件或I/O资源提供了一个唯一的标识符,通常是一个非负整数。
- 文件描述符表:每个进程都有自己的文件描述符表,这是一个内核数据结构,用于跟踪进程打开的所有文件和I/O资源。
- 系统调用:文件描述符通常通过系统调用如
open、read、write、close等进行操作。open调用返回一个新的文件描述符,read和write使用文件描述符来读取或写入数据,而close用于释放文件描述符。 - 标准流:Linux为标准输入(stdin)、标准输出(stdout)和标准错误(stderr)分别分配了文件描述符0、1和2。
- 缓冲机制:Linux内核可能会对通过文件描述符进行的I/O操作使用缓冲机制,以提高性能和减少实际的磁盘I/O操作。
- 错误处理:当系统调用失败时,会返回-1,并且全局变量
errno被设置为表示错误的特定值。 - 多路复用:文件描述符可以用于I/O多路复用机制,如
select、poll和epoll,允许进程同时监控多个文件描述符上的I/O状态。 - 继承性:当创建新进程时,子进程会继承父进程的文件描述符表中的文件描述符,除非它们在子进程中被显式地关闭。
- 重定向:文件描述符可以通过
dup、dup2等函数进行重定向,允许将一个文件描述符的引用复制到另一个文件描述符上。 - 文件锁:文件描述符可以用于对文件加锁,以控制对文件的并发访问。
6.12 缓冲区机制
缓冲区机制是C标准库中用于提高I/O性能的一种技术。缓冲区可以减少用户态和内核态之间的切换次数,从而提高性能。
- 全缓冲:对于文件,通常是全缓冲的。这意味着数据会先写入缓冲区,当缓冲区满或文件关闭时,数据才会被写入文件。
- 行缓冲:对于终端输入输出,通常是行缓冲的。这意味着数据会在遇到换行符时被写入文件。
- 无缓冲:对于标准错误输出,通常是无缓冲的。这意味着数据会立即被写入文件。
实际应用案例
6.13 文件拷贝示例
下面是一个使用库函数和系统调用实现文件拷贝的示例。该示例展示了如何结合使用库函数和系统调用来完成文件操作。
示例代码:
#include <stdio.h>
#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
#include <string.h>int copy_file(const char *src_path, const char *dst_path) {// 使用 fopen 打开源文件FILE *src_file = fopen(src_path, "rb");if (src_file == NULL) {fprintf(stderr, "打开源文件失败: %s\n", strerror(errno));return 1;}// 使用 open 系统调用打开目标文件int dst_fd = open(dst_path, O_WRONLY | O_CREAT | O_TRUNC, 0644);if (dst_fd == -1) {fprintf(stderr, "打开目标文件失败: %s\n", strerror(errno));fclose(src_file);return 1;}char buffer[1024];size_t bytes_read;while ((bytes_read = fread(buffer, 1, sizeof(buffer), src_file)) > 0) {if (ferror(src_file)) {fprintf(stderr, "读取源文件失败: %s\n", strerror(errno));fclose(src_file);close(dst_fd);return 1;}if (write(dst_fd, buffer, bytes_read) == -1) {perror("写入目标文件失败");fclose(src_file);close(dst_fd);return 1;}}if (ferror(src_file)) {fprintf(stderr, "读取源文件失败: %s\n", strerror(errno));fclose(src_file);close(dst_fd);return 1;}if (fclose(src_file) != 0) {fprintf(stderr, "关闭源文件失败: %s\n", strerror(errno));close(dst_fd);return 1;}if (close(dst_fd) == -1) {perror("关闭目标文件失败");return 1;}return 0;
}int main() {const char *src_path = "source.txt";const char *dst_path = "destination.txt";if (copy_file(src_path, dst_path) == 0) {printf("文件复制成功\n");} else {printf("文件复制失败\n");}return 0;
}
文件操作的底层原理
6.14 文件描述符与文件信息区
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字、文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是由系统声明的,取名 FILE。
每当打开一个文件的时候,系统会根据文件的情况自动创建一个 FILE 结构的变量,并填充其中的信息,使用者不必关心细节。一般都是通过一个 FILE 的指针来维护这个 FILE 结构的变量,这样使用起来更加方便。
6.15 文件指针的位置
文件指针可以指向文件的任意位置,通常用来记录下一次读取或写入的位置。可以通过一些函数来移动文件指针的位置,如 fseek 函数和 rewind 函数。可以通过一些函数来获取当前文件指针的位置,如 ftell 函数。
6.16 文件的打开和关闭
在使用文件之前应该打开文件,使用完之后应该关闭文件。ANSI C 规定使用 fopen 来打开文件,用 fclose 来关闭文件。
文件打开方式 mode 参数说明:
"r":只读模式,打开一个已经存在的文本文件,用于输入。"w":只写模式,打开一个文本文件用于输出,如果文件已存在则清空原有内容,如果文件不存在则创建新文件。"a":追加模式,打开一个文本文件用于在文件尾部追加数据,如果文件不存在则创建新文件。"rb":只读模式,打开一个二进制文件用于输入。"wb":只写模式,打开一个二进制文件用于输出,如果文件已存在则清空原有内容,如果文件不存在则创建新文件。"ab":追加模式,打开一个二进制文件用于在文件尾部追加数据,如果文件不存在则创建新文件。"r+":读写模式,打开一个文本文件用于读写,文件必须已存在。"w+":读写模式,打开一个文本文件用于读写,如果文件已存在则清空原有内容,如果文件不存在则创建新文件。"a+":读写模式,打开一个文本文件用于读写,文件不存在则创建新文件,所有写入操作都追加到文件尾部。"rb+":读写模式,打开一个二进制文件用于读写,文件必须已存在。"wb+":读写模式,打开一个二进制文件用于读写,如果文件已存在则清空原有内容,如果文件不存在则创建新文件。"ab+":读写模式,打开一个二进制文件用于读写,文件不存在则创建新文件,所有写入操作都追加到文件尾部。
6.17 文件的读写操作
文件的读写操作可以通过一系列函数来完成,如 fread、fwrite、fgetc、fputc、fgets、fputs 等。这些函数通常使用缓冲机制来提高性能。
6.18 文件定位
文件定位可以通过 fseek 函数来实现,该函数允许移动文件指针到文件中的任意位置。ftell 函数可以获取文件指针的当前位置。
fseek 函数参数说明:
stream:指向FILE结构的指针。offset:偏移量,可以是正数或负数。whence:定位基准点,可以是SEEK_SET(文件开头)、SEEK_CUR(当前文件位置)或SEEK_END(文件末尾)。
示例代码:
#include <stdio.h>
#include <errno.h>
#include <string.h>int main() {// 打开文件FILE *file = fopen("example.txt", "r+");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}// 移动文件指针到文件开头if (fseek(file, 0, SEEK_SET) != 0) {fprintf(stderr, "移动文件指针失败: %s\n", strerror(errno));fclose(file);return 1;}// 获取文件指针的当前位置long position = ftell(file);if (position == -1) {fprintf(stderr, "获取文件指针位置失败: %s\n", strerror(errno));fclose(file);return 1;}printf("文件指针位置: %ld\n", position);// 移动文件指针到文件末尾if (fseek(file, 0, SEEK_END) != 0) {fprintf(stderr, "移动文件指针失败: %s\n", strerror(errno));fclose(file);return 1;}// 获取文件指针的当前位置position = ftell(file);if (position == -1) {fprintf(stderr, "获取文件指针位置失败: %s\n", strerror(errno));fclose(file);return 1;}printf("文件指针位置: %ld\n", position);fclose(file);return 0;
}
6.19 文件错误处理
在进行文件操作时,必须注意处理可能出现的错误。可以使用 ferror 和 clearerr 函数来帮助诊断和清除错误状态。
示例代码:
#include <stdio.h>
#include <errno.h>
#include <string.h>int main() {// 打开文件FILE *file = fopen("nonexistent.txt", "r");if (file == NULL) {fprintf(stderr, "打开文件失败: %s\n", strerror(errno));return 1;}char buffer[100];size_t bytes_read;// 读取文件bytes_read = fread(buffer, 1, sizeof(buffer), file);if (ferror(file)) {fprintf(stderr, "读取文件失败: %s\n", strerror(errno));fclose(file);return 1;}// 清除错误状态clearerr(file);fclose(file);return 0;
}
文件映射
文件映射是一种高效的数据处理方法,它将文件内容直接映射到进程的虚拟地址空间,使得对文件的操作就像对内存的操作一样简单。文件映射通常通过 mmap 函数来实现。
6.20 使用 mmap 进行文件映射
mmap 函数可以将文件或其他对象映射到内存,映射的内存区域可以直接被读写。
示例代码:
#include <stdio.h>
#include <fcntl.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>int main() {// 打开文件int fd = open("largefile.dat", O_RDONLY);if (fd == -1) {perror("打开文件失败");return 1;}// 获取文件大小struct stat st;if (fstat(fd, &st) == -1) {perror("获取文件大小失败");close(fd);return 1;}// 映射文件到内存void *addr = mmap(NULL, st.st_size, PROT_READ, MAP_PRIVATE, fd, 0);if (addr == MAP_FAILED) {perror("映射文件失败");close(fd);return 1;}// 处理映射内存// ...// 解映射内存if (munmap(addr, st.st_size) == -1) {perror("解映射内存失败");close(fd);return 1;}// 关闭文件if (close(fd) == -1) {perror("关闭文件失败");return 1;}return 0;
}
总结
本文详细介绍了C语言文件操作中的库函数和系统调用,解释了它们的工作原理、区别和联系,并通过实际示例展示了如何使用这些函数。通过本文的学习,读者应能全面理解C语言文件操作的底层机制,为编写高效、可靠的程序提供有力支持。
希望本文能够帮助读者深入理解和应用C语言中的文件操作技术。如果您有任何进一步的问题或建议,请随时留言交流。
相关文章:
深入C语言文件操作:从库函数到系统调用
引言 文件操作是编程中不可或缺的一部分,尤其在C语言中,文件操作不仅是处理数据的基本手段,也是连接程序与外部世界的重要桥梁。C语言提供了丰富的库函数来处理文件,如 fopen、fclose、fread、fwrite 等。然而,这些库…...
Java序列化
Java序列化 简单来说: 序列化是将对象的状态信息转换为可以存储或传输的形式(如字节序列)的过程。在 Java 中,通过序列化可以把一个对象保存到文件、通过网络传输到其他地方或者存储到数据库等。最直接的原因就是某些场景下需要…...
基坑表面位移沉降倾斜自动化监测 非接触式一体化解决机器视觉
基于变焦视觉位移监测仪的基坑自动化监测新方案是一种集成了光学、机械、电子、边缘计算、AI识别以及云平台软件等技术的自动化系统。该方案利用变焦机器视觉原理,结合特殊波段成像识别技术和无源靶标,实现了非接触式大空间、多断面、多测点的高精度水平…...
提升效率:精通Windows命令行的艺术
文章目录 引言1. 基本目录操作命令dir:列出目录内容cd:更改目录mkdir 和 rmdir:创建和删除目录 2. 文件操作命令copy:复制文件或目录move:移动或重命名文件/目录del:删除文件 3. 文件查看命令typeÿ…...
ESP32-S3-devKitC-1 点亮板上的WS2812 RGB LED
ESP32-S3-devKitC-1 板上自带了一个RGB LED,型号为 WS2812。 RGB LED 在板上的位置如下图所示。 为了点亮这个WS2812,需要确定这颗RGB LED连接到哪个GPIO上了。 下面是确定GPIO管脚的过程: 1、根据原理图 2、根据PCB布局图: 程…...
python调用matlab函数(内置 + 自定义) —— 安装matlab.engine
文章目录 一、简介二、安装matlab.engine2.1、基于 CMD 安装2.2、基于 MATLAB 安装(不建议) 三、python调用matlab函数(内置 自定义) 一、简介 matlab.engine(MATLAB Engine API for Python):…...
CAD c# 生成略缩图预览
代码如下: using (Transaction tr currentdb.TransactionManager.StartTransaction()){//当前数据库开启事务using (Database tempdb new Database(false, true)) //创建临时数据库(两个参数:是否创建符号表,不与当前文档关联){try{Bitmap …...
端点鉴别、安全电子邮件、TLS
文章目录 端点鉴别鉴别协议ap 1.0——发送者直接发送一个报文表明身份鉴别协议ap 2.0——ap1.0 的基础上,接收者对报文的来源IP地址进行鉴别鉴别协议ap 3.0——使用秘密口令,口令为鉴别者和被鉴别者之间共享的秘密鉴别协议ap 3.1——对秘密口令进行加密&…...
汽车电子元件的可靠性保障:AEC-Q102认证
AEC-Q102标准的起源与价值 随着汽车电子系统的日益复杂,电子器件必须能够在极端的温度、湿度、振动和电磁干扰等恶劣条件下保持性能。AEC-Q102标准由汽车电子委员会(AEC)制定,专门针对LED、激光二极管和光电二极管等光电器件&…...
主成分分析法大全(包括stata+matlab)
数据简介:主成分分析(Principal Component Analysis,PCA), 是一种统计方法。通过正交变换将一组可能存在相关性的变量转换为一组线性不相关的变量,转换后的这组变量叫主成分。在实际课题中,为了…...
ubuntu+ros新手笔记(五):初探anaconda+cuda+pytorch
深度学习三件套:初探anacondacudapytorch 系统ubuntu22.04 1.初探anaconda 1.1 安装 安装过程参照【详细】Ubuntu 下安装 Anaconda 1.2 创建和删除环境 创建新环境 conda create -n your_env_name pythonx.x比如我创建了一个名为“py312“的环境 conda cre…...
C++ List(双向链表)
是一个线性链表结构,它的数据由若干个节点构成,每一个节点都包括一个 信息块(即实际存储的数据)、一个前驱指针和一个后驱指针。它无需分配指定 的内存大小且可以任意伸缩,这是因为它存储在非连续的内存空间中&#…...
ASP.NET|日常开发中读写TXT文本详解
ASP.NET|日常开发中读写TXT文本详解 前言一、读取 TXT 文本1.1 使用StreamReader类 二、写入 TXT 文本2.1 使用StreamWriter类 三、文件编码问题3.1 常见编码格式 四、错误处理和性能考虑4.1 错误处理4.2 性能考虑 结束语优质源码分享 ASP.NET|日常开发中…...
【机器学习】在不确定的光影中:机器学习与概率论的心灵共舞
文章目录 概率与统计基础:解锁机器学习的数据洞察之门前言一、概率论基础1.1 概率的基本概念与性质1.1.1 概率的定义1.1.2 样本空间与事件1.1.3 互斥事件与独立事件1.1.4 概率的计算方法 1.2 条件概率与独立性1.2.1 条件概率1.2.2 独立事件 1.3 随机变量1.3.1 随机变…...
【论文笔记】Editing Models with Task Arithmetic
🍎个人主页:小嗷犬的个人主页 🍊个人网站:小嗷犬的技术小站 🥭个人信条:为天地立心,为生民立命,为往圣继绝学,为万世开太平。 基本信息 标题: Editing Models with Task…...
ESP32外设学习部分--UART篇
前言 在我们学习嵌入式的过程中,uart算是我们用的非常多的一个外设了,我们可以用串口打印信息,也可以用于设备通信,总之串口的作用非常多,我们也非常有必要熟练地去掌握这个外设。 uart的配置 uart的参数配置 uart_…...
ssm-day04 mybatis
mybatis是一个持久层框架,针对的是JDBC的优化 简化数据库操作,能进行单表、多表操作,在这个框架下,需要我们自己写SQL语句 Mapper接口和MapperXML文件就相当于Dao和Dao层的实现 通常将xml文件放在resources包下 ,放在…...
es中段是怎么合并的
文章目录 1. 段合并的背景2. 合并的方式2.1TieredMergePolicy 的层次结构2.2 层次的基本规则2.3 如何理解层次(tier)2.4. 合并过程中的层次示例2.5. TieredMergePolicy 的优势2.6. 小结 3. 合并过程中的优化4. 合并的性能考虑5. 使用 API 手动合并6. 合并…...
5、可暂停的线程控制模型
一、需求 在做播放器的时候,很多的模块会创建一个线程,然后在这个线程上跑单独的功能,同时,需要对这个线程进行控制,比如暂停,继续等,如播放器的解码,解封装等,都需要对…...
sql优化--mysql隐式转换
sql隐式转换 在SQL中,隐式转换是数据库自动进行的类型转换,隐式转换可以帮助我们处理不同类型的数据。 比如,数据表的字段是字符串类型的,传入一个整型的数据,也能够运行sql。 sql隐式转换的弊端 sql隐式转换&…...
第19节 Node.js Express 框架
Express 是一个为Node.js设计的web开发框架,它基于nodejs平台。 Express 简介 Express是一个简洁而灵活的node.js Web应用框架, 提供了一系列强大特性帮助你创建各种Web应用,和丰富的HTTP工具。 使用Express可以快速地搭建一个完整功能的网站。 Expre…...
ssc377d修改flash分区大小
1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...
前端导出带有合并单元格的列表
// 导出async function exportExcel(fileName "共识调整.xlsx") {// 所有数据const exportData await getAllMainData();// 表头内容let fitstTitleList [];const secondTitleList [];allColumns.value.forEach(column > {if (!column.children) {fitstTitleL…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
【Linux】Linux 系统默认的目录及作用说明
博主介绍:✌全网粉丝23W,CSDN博客专家、Java领域优质创作者,掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围:SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物…...
Linux nano命令的基本使用
参考资料 GNU nanoを使いこなすnano基础 目录 一. 简介二. 文件打开2.1 普通方式打开文件2.2 只读方式打开文件 三. 文件查看3.1 打开文件时,显示行号3.2 翻页查看 四. 文件编辑4.1 Ctrl K 复制 和 Ctrl U 粘贴4.2 Alt/Esc U 撤回 五. 文件保存与退出5.1 Ctrl …...
C++ 设计模式 《小明的奶茶加料风波》
👨🎓 模式名称:装饰器模式(Decorator Pattern) 👦 小明最近上线了校园奶茶配送功能,业务火爆,大家都在加料: 有的同学要加波霸 🟤,有的要加椰果…...
适应性Java用于现代 API:REST、GraphQL 和事件驱动
在快速发展的软件开发领域,REST、GraphQL 和事件驱动架构等新的 API 标准对于构建可扩展、高效的系统至关重要。Java 在现代 API 方面以其在企业应用中的稳定性而闻名,不断适应这些现代范式的需求。随着不断发展的生态系统,Java 在现代 API 方…...
tauri项目,如何在rust端读取电脑环境变量
如果想在前端通过调用来获取环境变量的值,可以通过标准的依赖: std::env::var(name).ok() 想在前端通过调用来获取,可以写一个command函数: #[tauri::command] pub fn get_env_var(name: String) -> Result<String, Stri…...
算法打卡第18天
从中序与后序遍历序列构造二叉树 (力扣106题) 给定两个整数数组 inorder 和 postorder ,其中 inorder 是二叉树的中序遍历, postorder 是同一棵树的后序遍历,请你构造并返回这颗 二叉树 。 示例 1: 输入:inorder [9,3,15,20,7…...
