Linux文件编程详解

news2024/11/26 1:56:11

Linux文件编程详解

在Ubuntu(Linux)系统下进行文件操作涉及一系列的系统调用,这些调用是基于Unix风格的文件操作API。这些操作包括打开或创建文件、从文件中读取数据、向文件中写入数据、移动文件指针以及关闭文件。以下是这些函数的详细介绍和实际应用示例。

1. 创建和打开文件 (open)

open 函数简介

open 函数用于打开一个现有文件或创建一个新文件。当配合 O_CREAT 标志使用时,它可以创建一个新的文件。

函数原型
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

int open(const char *pathname, int flags, ... /* mode_t mode */);
  • 参数说明

  • pathname: 要打开或创建的文件的路径。
  • flags: 控制文件打开或创建方式的标志,可以是:
    • O_RDONLY:只读打开。
    • O_WRONLY:只写打开。
    • O_RDWR:读写打开。
    • O_CREAT:如果文件不存在,则创建它。
    • O_EXCL:与 O_CREAT 一起使用时,如果文件已存在,则返回错误。
    • O_TRUNC:如果文件已存在并且是以写入方式打开,则清空文件。
  • mode (可选参数):指定新文件的权限。这个参数是在文件创建时设置的,通常与 umask 的默认设置一起使用,例如 06440755 等。
 示例代码:创建文件
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <unistd.h>

int main() {
    int fd;  // 文件描述符

    // 创建文件,如果文件不存在。设置文件权限为 0644 (rw-r--r--)
    fd = open("newfile.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644);
    if (fd == -1) {
        perror("创建文件失败");
        exit(EXIT_FAILURE);
    }

    printf("文件 'newfile.txt' 已成功创建\n");

    // 对文件进行写操作等
    // ...

    // 完成操作后,关闭文件
    close(fd);
    return 0;
}

2. 写入文件 (write)

write 函数将数据写入文件。

函数原型
#include <unistd.h>

ssize_t write(int fd, const void *buf, size_t count);
  • fd:文件描述符,由 open 返回。
  • buf:指向数据缓冲区的指针。
  • count:要写入的字节数。
 示例代码:写入数据
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>

int main() {
    int fd;
    fd = open("testfile.txt", O_WRONLY|O_CREAT|O_TRUNC, 0644);
    if (fd == -1) {
        perror("打开文件失败");
        exit(EXIT_FAILURE);
    }

    char *data = "Hello, Ubuntu 文件编程!";
    // 将字符串数据写入文件
    if (write(fd, data, strlen(data)) == -1) {
        perror("写入文件失败");
        close(fd);
        exit(EXIT_FAILURE);
    }

    printf("数据写入成功\n");

    // 写入完毕,关闭文件
    close(fd);
    return 0;
}

 3. 读取文件(read)

参数说明

  • fd:文件描述符,表示要读取数据的文件。这个文件描述符通常是通过 open 系统调用获取的。
  • buf:指向数据缓冲区的指针,用于存储从文件中读取的数据。
  • count:指定要读取的最大字节数。实际读取的字节数可能少于这个值,比如文件剩余内容不足 count 字节时。

返回值

  • 返回读取的字节数,如果文件已到达末尾,则返回 0。
  • 如果发生错误,则返回 -1,并设置 errno 以指示错误类型。
函数原型:
#include <unistd.h>

ssize_t read(int fd, void *buf, size_t count);

示例代码: 

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>

int main() {
    int fd;  // 文件描述符
    ssize_t bytes_read;  // 实际读取的字节数
    char buffer[1024];  // 数据缓冲区

    // 打开文件
    fd = open("example.txt", O_RDONLY);
    if (fd == -1) {
        perror("打开文件失败");
        exit(EXIT_FAILURE);
    }

    // 从文件中读取数据
    bytes_read = read(fd, buffer, sizeof(buffer) - 1);  // 保留一个位置用于放置字符串终止符'\0'
    if (bytes_read == -1) {
        perror("读取文件失败");
        close(fd);
        exit(EXIT_FAILURE);
    }

    buffer[bytes_read] = '\0';  // 确保读取的内容被正确终止以便作为字符串处理

    // 打印读取的内容
    printf("读取到的内容:\n%s\n", buffer);

    // 关闭文件
    close(fd);
    return 0;
}

4. 关闭文件 (close)

close 函数用于关闭一个打开的文件描述符,释放系统资源。

函数原型
#include <unistd.h>

int close(int fd);

fd:要关闭的文件描述符。

关闭文件的重要性 

关闭文件是必要的,因为打开的文件描述符是有限的资源。每个进程能打开的文件数量是有限的,未及时关闭文件可能导致资源泄漏,影响程序性能或导致程序异常。

5. 移动文件 (lseek)

lseek 函数用于移动文件的读/写指针到指定位置。

函数原型
#include <sys/types.h>
#include <unistd.h>

off_t lseek(int fd, off_t offset, int whence);
  • fd:文件描述符。
  • offset:偏移量,可为负值。
  • whence:起始位置(SEEK_SET 从文件开始,SEEK_CUR 从当前位置,SEEK_END 从文件末尾)。
 移动文件光标使用 lseek 系统调用。下面是一个示例代码:
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>

#define FILENAME "example.txt"

int main() {
    // 以只读模式打开文件 example.txt
    int fd = open(FILENAME, O_RDONLY);
    if (fd == -1) {
        // 打开文件失败,输出错误信息并退出程序
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 将文件光标移动到文件末尾,并获取文件大小
    off_t offset = lseek(fd, 0, SEEK_END);
    if (offset == -1) {
        // lseek操作失败,输出错误信息并关闭文件描述符
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 打印文件大小
    printf("文件大小: %ld 字节\n", offset);

    // 将文件光标移动回文件开头
    if (lseek(fd, 0, SEEK_SET) == -1) {
        // lseek操作失败,输出错误信息并关闭文件描述符
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 关闭文件描述符
    close(fd);
    return 0;
}

6. 计算文件大小(lseek)

计算文件大小可以使用 lseek 系统调用将文件光标移动到文件末尾,然后获取当前位置的偏移量。下面是一个示例代码:
#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>

#define FILENAME "example.txt"

int main() {
    // 以只读模式打开文件 example.txt
    int fd = open(FILENAME, O_RDONLY);
    if (fd == -1) {
        // 打开文件失败,输出错误信息并退出程序
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 将文件光标移动到文件末尾,并获取文件大小
    off_t file_size = lseek(fd, 0, SEEK_END);
    if (file_size == -1) {
        // lseek操作失败,输出错误信息并关闭文件描述符
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 打印文件大小
    printf("文件大小: %ld 字节\n", file_size);

    // 关闭文件描述符
    close(fd);
    return 0;
}

7打开/创建文件打开/创建文件

7.cp文件的实现

mian函数参数的用法
#include <stdio.h>
 
int main(int argc,char **argv)
{
        printf("total params:%d\n",argc);
        printf("No.1 params :%s\n",argv[0]);
        printf("No.2 params :%s\n",argv[1]);
        printf("No.3 params :%s\n",argv[2]);
 
        return 0;
}

argc :代表的是 ./a.out argc argv 这三个参数的个数

argv[0] :代表第一个参数./a.out

argv[1] :代表第二个参数 argc

argv[2] :代表第二个参数 argv

由此可见argv是数组的数组。

思路:1.打开源文件src.c

​ 2.读src到buf

​ 3.打开/创建目标文件des.c

​ 4.将buf写入des.c

​ 5.close两个文件

#include <sys/types.h>   // 引入类型定义,如pid_t等
#include <sys/stat.h>    // 引入文件状态定义,如S_IRUSR等
#include <fcntl.h>       // 引入文件控制定义,如O_RDWR等
#include <stdio.h>       // 引入标准输入输出库,用于printf等函数
#include <unistd.h>      // 引入POSIX操作系统API,如read、write等
#include <string.h>      // 引入字符串操作库,用于strlen等函数
#include <stdlib.h>      // 引入标准库,用于malloc、exit等函数

int main(int argc, char **argv)
{
    int fdSrc;  // 源文件的文件描述符
    int fdDes;  // 目标文件的文件描述符

    char *readBuf=NULL;  // 用于存储从源文件读取的数据

    if(argc != 3){  // 检查命令行参数数量是否正确
        printf("pararm error\n");  // 参数错误时打印错误信息
        exit(-1);  // 错误退出程序
    }

    fdSrc = open(argv[1], O_RDWR);  // 以读写方式打开源文件
    int size = lseek(fdSrc, 0, SEEK_END);  // 定位到源文件的末尾,获取文件大小
    lseek(fdSrc, 0, SEEK_SET);  // 重新定位到文件开头,准备读取数据

    readBuf = (char *)malloc(sizeof(char) * size + 8);  // 分配足够的内存以存储文件数据,额外加8字节防止溢出

    int n_read = read(fdSrc, readBuf, size);  // 从源文件读取数据到缓冲区

    fdDes = open(argv[2], O_RDWR | O_CREAT | O_TRUNC, 0600);  // 以读写方式打开或创建目标文件,若文件存在则长度截为0

    int n_write = write(fdDes, readBuf, strlen(readBuf));  // 将读取的数据写入目标文件,使用strlen确保只写入有效字符串长度

    close(fdSrc);  // 关闭源文件
    close(fdDes);  // 关闭目标文件

    return 0;  // 程序正常结束
}

 8.文件编程修改程序的配置文件demo

#include <sys/types.h>  // 引入用于系统调用的基本数据类型
#include <sys/stat.h>   // 引入用于文件属性操作的定义
#include <fcntl.h>      // 引入用于文件控制的函数定义
#include <stdio.h>      // 引入标准输入输出库
#include <unistd.h>     // 引入POSIX操作系统API
#include <string.h>     // 引入字符串操作函数
#include <stdlib.h>     // 引入标准库函数

int main(int argc, char **argv)
{
    int fdSrc;  // 定义源文件的文件描述符

    char *readBuf = NULL;  // 定义读取数据的缓冲区指针

    if(argc != 2){  // 检查输入参数是否为2个,不正确则报错
        printf("parameter error\n");  // 打印参数错误信息
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }
    
    // 以读写模式打开源文件,文件描述符保存在fdSrc中
    fdSrc = open(argv[1], O_RDWR);
    if (fdSrc == -1) {  // 检查文件是否成功打开
        perror("Error opening file");  // 打印错误信息
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 使用lseek获取文件大小
    int size = lseek(fdSrc, 0, SEEK_END);
    if (size == -1) {  // 检查是否成功获取文件大小
        perror("Error seeking file");  // 打印错误信息
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 将文件指针重新定位到文件开始
    if (lseek(fdSrc, 0, SEEK_SET) == -1) {
        perror("Error re-seeking file");  // 打印错误信息
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 分配内存以存储文件数据,多分配8字节防止溢出
    readBuf = (char *)malloc(sizeof(char) * size + 8);
    if (readBuf == NULL) {  // 检查内存分配是否成功
        perror("Memory allocation failed");  // 打印错误信息
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 从文件中读取数据到缓冲区
    int n_read = read(fdSrc, readBuf, size);
    if (n_read == -1) {  // 检查读取是否成功
        perror("Error reading file");  // 打印错误信息
        free(readBuf);  // 释放内存
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 查找特定的字符串"LENG=",准备修改其后的值
    char *p = strstr(readBuf, "LENG=");
    if (p == NULL) {  // 检查是否找到指定字符串
        printf("not found\n");  // 打印未找到信息
        free(readBuf);  // 释放内存
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 修改找到字符串后的值
    p += strlen("LENG=");
    *p = '5';  // 设置新的值为'5'

    // 将文件指针重新定位到文件开始
    if (lseek(fdSrc, 0, SEEK_SET) == -1) {
        perror("Error re-seeking file");  // 打印错误信息
        free(readBuf);  // 释放内存
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 将修改后的缓冲区内容写回文件
    int n_write = write(fdSrc, readBuf, strlen(readBuf));
    if (n_write == -1) {  // 检查写入是否成功
        perror("Error writing file");  // 打印错误信息
        free(readBuf);  // 释放内存
        close(fdSrc);  // 关闭文件
        exit(EXIT_FAILURE);  // 退出程序,返回失败标志
    }

    // 释放内存并关闭文件描述符
    free(readByf);
    close(fdSrc);

    return 0;  // 返回成功
}

9.写一个整数到文件和写结构体数组到文件

示例程序 1: 写入和读取单个整数
#include <sys/types.h>  // 引入用于系统调用的基本数据类型
#include <sys/stat.h>   // 引入用于文件属性操作的定义
#include <fcntl.h>      // 引入用于文件控制的函数定义
#include <stdio.h>      // 引入标准输入输出库
#include <unistd.h>     // 引入POSIX操作系统API
#include <string.h>     // 引入字符串操作函数
#include <stdlib.h>     // 引入标准库函数

int main()
{
	int fd;  // 文件描述符
	
	int data = 100;  // 要写入文件的整数
	int data2 = 0;  // 用于读取文件数据的整数

	fd = open("./file1",O_RDWR);  // 打开文件以读写方式

	int n_write = write(fd,&data,sizeof(int));  // 将整数data写入文件

	lseek(fd,0,SEEK_SET);  // 重置文件指针到文件开始处

	int n_read = read(fd, &data2, sizeof(int));  // 从文件读取整数到data2
	
	printf("read %d \n",data2);  // 打印读取的整数
	close(fd);  // 关闭文件描述符

	return 0;  // 程序正常退出
}
示例程序 2: 写入和读取单个结构体
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>

struct Test  // 定义一个结构体,包含一个整数和一个字符
{
	int a;
	char c;
};

int main()
{
	int fd;  // 文件描述符
	
	struct Test data = {100,'a'};  // 初始化结构体变量data
	struct Hest data2;  // 用于读取文件数据的结构体变量

	fd = open("./file1",O_RDWR);  // 打开文件以读写方式

	int n_write = write(fd,&data,sizeof(struct Test));  // 将结构体data写入文件

	lseek(fd,0,SEEK_SET);  // 重置文件指针到文件开始处

	int n_read = read(fd, &data2, sizeof(struct Test));  // 从文件读取结构体到data2
	
	printf("read %d,%c \n",data2.a,data2.c);  // 打印读取的结构体内容
	close(fd);  // 关闭文件描述符

	return 0;  // 程序正常退出
}
示例程序 3: 写入和读取结构体数组
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <stdlib.h>

struct Test  // 定义一个结构体,包含一个整数和一个字符
{
	int a;
	char c;
};

int main()
{
	int fd;  // 文件描述符
	
	struct Test data[2] = {{100,'a'},{101,'b'}};  // 初始化结构体数组
	struct Test data2[2];  // 用于读取文件数据的结构体数组

	fd = open("./file1",O_RDWR);  // 打开文件以读写方式

	int n_write = write(fd,&data,sizeof(struct Test)*2);  // 将结构体数组data写入文件

	lseek(fd,0,SEEK_SET);  // 重置文件指针到文件开始处

	int n_read = read(fd, &data2, sizeof(struct Test)*2);  // 从文件读取结构体数组到data2
	
	printf("read %d,%c \n",data2[0].a,data2[0].c);  // 打印读取的第一个结构体内容
	printf("read %d,%c \n",data2[1].a,data2[1].c);  // 打印读取的第二个结构体内容
	close(fd);  // 关闭文件描述符

	return 0;  // 程序正常退出
}

10.open和fopen区别

openfopen 是用于文件操作的两个常见函数,它们存在一些关键差异,主要体现在它们所属的库和API风格、功能、以及使用场景上。

 1. 来源

从来源的角度看,两者能很好的区分开,这也是两者最显而易见的区别:

  • open是UNIX系统调用函数(包括LINUX等),返回的是文件描述符(File Descriptor),它是文件在文件描述符表里的索引。
  • fopen是ANSIC标准中的C语言库函数,在不同的系统中应该调用不同的内核api。返回的是一个指向文件结构的指针。 
     PS:从来源来看,两者是有千丝万缕的联系的,毕竟C语言的库函数还是需要调用系统API实现的。
2. 移植性

这一点从上面的来源就可以推断出来,`fopen`是C标准函数,因此拥有良好的移植性;而`open`是UNIX系统调用,移植性有限。如windows下相似的功能使用API函数`CreateFile`。

3. 适用范围
  • open返回文件描述符,而文件描述符是UNIX系统下的一个重要概念,UNIX下的一切设备都是以文件的形式操作。如网络套接字、硬件设备等。当然包括操作普通正规文件(Regular File)。
  • fopen是用来操纵普通正规文件(Regular File)的。
4. 文件IO层次

如果从文件IO的角度来看,前者属于低级IO函数,后者属于高级IO函数。低级和高级的简单区分标准是:谁离系统内核更近。低级文件IO运行在内核态,高级文件IO运行在用户态。

5. 缓冲
  1. 缓冲文件系统 
    缓冲文件系统的特点是:在内存开辟一个“缓冲区”,为程序中的每一个文件使用;当执行读文件的操作时,从磁盘文件将数据先读入内存“缓冲区”,装满后再从内存“缓冲区”依此读出需要的数据。执行写文件的操作时,先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。由此可以看出,内存“缓冲区”的大小,影响着实际操作外存的次数,内存“缓冲区”越大,则操作外存的次数就少,执行速度就快、效率高。一般来说,文件“缓冲区”的大小随机器 而定。fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind等。
  2. 非缓冲文件系统 
    缓冲文件系统是借助文件结构体指针来对文件进行管理,通过文件指针来对文件进行访问,既可以读写字符、字符串、格式化数据,也可以读写二进制数据。非缓冲文件系统依赖于操作系统,通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件,但效率高、速度快,由于ANSI标准不再包括非缓冲文件系统,因此建议大家最好不要选择它。open, close, read, write, getc, getchar, putc, putchar等。

一句话总结一下,就是open无缓冲,fopen有缓冲。前者与readwrite等配合使用, 后者与fread,fwrite等配合使用。

使用fopen函数,由于在用户态下就有了缓冲,因此进行文件读写操作的时候就减少了用户态和内核态的切换(切换到内核态调用还是需要调用系统调用API:readwrite);而使用open函数,在文件读写时则每次都需要进行内核态和用户态的切换;表现为,如果顺序访问文件,fopen系列的函数要比直接调用open系列的函数快;如果随机访问文件则相反。

这样一总结梳理,相信大家对于两个函数及系列函数有了一个更全面清晰的认识,也应该知道在什么场合下使用什么样的函数更合适,效率更高。

 使用 open 示例:
#include <fcntl.h>    // 引入文件控制的头文件
#include <unistd.h>   // 引入POSIX操作系统API
#include <sys/stat.h> // 引入文件状态的头文件
#include <stdio.h>    // 引入标准输入输出库
#include <stdlib.h>   // 引入标准库,用于使用exit等函数

int main() {
    // 以写入、创建、截断方式打开文件,文件权限为用户读写
    int fd = open("example.dat", O_WRONLY | O_CREAT | O_TRUNC, S_IRUSR | S_IWUSR);
    if (fd == -1) {
        perror("Failed to open file");  // 打开文件失败时输出错误信息
        exit(EXIT_FAILURE);             // 退出程序
    }

    const char *text = "Hello, world!";  // 定义要写入的字符串
    if (write(fd, text, 13) != 13) {     // 写入字符串到文件
        perror("Failed to write");       // 写入失败时输出错误信息
        close(fd);                       // 关闭文件描述符
        exit(EXIT_FAILURE);              // 退出程序
    }

    close(fd);  // 成功写入后关闭文件描述符
    return 0;
}
使用 fopen 示例:
#include <stdio.h>   // 引入标准输入输出库
#include <stdlib.h>  // 引入标准库,用于使用exit等函数

int main() {
    // 以写入模式打开文件
    FILE *fp = fopen("example.txt", "w");
    if (fp == NULL) {
        perror("Failed to open file");  // 打开文件失败时输出错误信息
        exit(EXIT_FAILURE);             // 退出程序
    }

    const char *integer = "Hello, world!"; // 定义要写入的字符串
    if (fprintf(fp, "%s", integer) < 0) {  // 使用fprintf将字符串写入文件
        perror("Failed to write");         // 写入失败时输出错误信息
        fclose(fp);                        // 关闭文件流
        exit(EXIT_FAILURE);                // 退出程序
    }

    fclose(fp);  // 成功写入后关闭文件流
    return 0;
}

11.标准C库打开C库打开创建文件读写文件光标移动

#include <stdio.h>   // 引入标准输入输出库,用于文件操作和基本的输入输出功能
#include <string.h>  // 引入字符串处理库,用于处理字符串,如计算字符串长度等

int main()
{
    FILE *fp;  // 声明一个 FILE 类型的指针,用于表示文件流
    char *str = "hello world";  // 声明并初始化一个字符串常量,将用于写入文件
    char readBuf[128] = {0};  // 声明一个字符数组作为读取缓冲区,并初始化所有元素为 0

    fp = fopen("./hello.txt","w+");  // 使用 fopen 函数以读写模式打开(如果不存在则创建)一个名为 chen.txt 的文件

    // 使用 fwrite 函数将字符串 str 写入文件
    // str: 要写入的数据的指针
    // sizeof(char)*strlen(str): 每个数据单元的大小乘以要写入的数据单元的数量,这里计算整个字符串的总大小
    // 1: 写入的数据块数量,这里写入一次整个字符串
    // fp: 目标文件流的指针
    fwrite(str, sizeof(char)*strlen(str), 1, fp);

    fseek(fp, 0, SEEK_SET);  // 使用 fseek 函数将文件内的位置指针重新定位到文件开头

    // 使用 fread 函数从文件中读取数据
    // readBuf: 用于接收数据的缓冲区的指针
    // sizeof(char)*strlen(str): 每个数据单元的大小乘以数据单元的数量,这里计算整个字符串的总大小
    // 1: 读取的数据块数量,这里读取一次整个字符串
    // fp: 源文件流的指针
    fread(readModule, sizeof(char)*strlen(str), 1, fp);
    
    printf("read data: %s\n", readBuf);  // 使用 printf 函数打印读取到的数据

    return 0;  // 程序正常结束,返回 0
}

 12.第一个程序:写入并读取字符串  第二个程序:写入并读取结构体

#include <stdio.h>  // 引入标准输入输出库
#include <string.h> // 引入字符串处理库

int main() {
    FILE *fp;  // 文件指针
    char *str = "hello world";  // 要写入文件的字符串
    char readBuf[128] = {0};  // 读取数据的缓冲区,初始化为0

    fp = fopen("./hello.txt", "w+");  // 以读写模式打开文件,如果文件不存在则创建
    if (fp == NULL) {  // 文件打开失败的处理
        perror("Failed to open file");
        return -1;
    }

    int nwrite = fwrite(str, sizeof(char) * strlen(str), 1, fp);  // 将字符串写入文件
    fseek(fp, 0, SEEK_SET);  // 重置文件指针到文件开头
    int nread = fread(readBuf, sizeof(char) * strlen(str), 100, fp);  // 从文件读取数据到缓冲区

    printf("read data: %s\n", readBuf);  // 打印读取的数据
    printf("read=%d, write = %d\n", nread, nwrite);  // 打印读取和写入的数据数量

    return 0;  // 程序结束
}
#include <stdio.h>    // 引入标准输入输出库
#include <sys/types.h> // 引入系统类型定义
#include <sys/stat.h>  // 引入文件状态控制库
#include <fcntl.h>     // 引入文件控制函数库
#include <unistd.h>    // 引入POSIX操作系统API
#include <string.h>    // 引入字符串处理库
#include <stdlib.h>    // 引入标准库函数

struct Test {
    int a;   // 结构体的整型成员
    char c;  // 结构体的字符成员
};

int main() {
    FILE *fp;  // 文件指针
    struct Test data = {100, 'a'};  // 初始化结构体实例
    struct Test data2;  // 用于存储从文件读取的结构体数据

    fp = fopen("./file1", "w+");  // 以读写模式打开文件
    if (fp == NULL) {  // 文件打开失败的处理
        perror("Failed to open file");
        return -1;
    }

    int n_write = fwrite(&data, sizeof(struct Test), 1, fp);  // 将结构体写入文件
    fseek(fp, 0, SEEK_SET);  // 重置文件指针到文件开头
    int n_read = fread(&data2, sizeof(struct Test), 1, fp);  // 从文件读取结构体数据

    printf("read %d,%c \n", data2.a, data2.c);  // 打印读取的结构体成员
    fclose(fp);  // 关闭文件

    return 0;  // 程序结束
}
第一个程序:向文件中写入字符串
#include <stdio.h>  // 引入标净输入输出库
#include <string.h> // 引入字符串处理库

int main()
{
    FILE *fp;  // 文件指针
    int i;  // 循环计数器
    char *str = "zlb hen shuai o!";  // 要写入文件的字符串
    int len = strlen(str);  // 计算字符串的长度

    fp = fopen("./test.txt", "w+");  // 以读写模式打开文件,如果文件不存在则创建
    if (fp == NULL) {  // 文件打开失败的处理
        perror("Failed to open file");
        return -1;
    }

    for (i = 0; i < len; i++) {  // 遍历字符串中的每个字符
        fputc(*str, fp);  // 将当前字符写入文件
        str++;  // 移动到字符串的下一个字符
    }
    fclose(fp);  // 关闭文件
    return 0;
}
第二个程序:从文件中读取字符串
#include <stdio.h>  // 引入标准输入输出库
#include <string.h> // 引入字符串处理库

int main()
{
    FILE *fp;  // 文件指针
    int i;  // 循环计数器
    char c;  // 用于存储读取的字符

    fp = fopen("./test.txt", "r");  // 以只读模式打开文件
    if (fp == NULL) {  // 文件打开失败的处理
        perror("Failed to open file");
        return -1;
    }

    while (!feof(fp)) {  // 判断文件是否结束
        c = fgetc(fp);  // 从文件中读取一个字符
        printf("%c", c);  // 打印读取的字符
    }
    fclose(fp);  // 关闭文件
    return 0;
}

 

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

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

相关文章

WPF/C#:在DataGrid中显示选择框

前言 在使用WPF的过程中可能会经常遇到在DataGrid的最前或者最后添加一列选择框的需求&#xff0c;今天跟大家分享一下&#xff0c;在自己的项目中是如何实现的。 整体实现效果如下&#xff1a; 如果对此感兴趣&#xff0c;可以接下来看具体实现部分。 实践 假设数据库中的…

LeetCode LCP 61. 气温变化趋势

别怕麻烦&#xff0c;模拟题有时候就是要多写一些条件&#xff08;或者你思维很活跃找出规律&#xff09;&#xff0c;代码如下&#xff1a; class Solution { public:int temperatureTrend(vector<int>& temperatureA, vector<int>& temperatureB) {int …

GraphQL vs REST:API设计的现代选择

随着技术的飞速发展&#xff0c;API&#xff08;应用程序接口&#xff09;设计成为了软件开发中不可或缺的一部分。REST&#xff08;Representational State Transfer&#xff09;和GraphQL作为两种主流的API设计风格&#xff0c;各自具有独特的优势和适用场景。本文将深入探讨…

audacity音频处理

1.安装 Audacity | Free Audio editor, recorder, music making and more! 添加OpenVINO插件: https://zhuanlan.zhihu.com/p/676542556 2.使用 2.1注意事项 1.编辑音频,点击左上方打开或导入都可以;打开视频文件则需要安装ffmpeg模块,打开后只显示视频中的音频信息; 2.编辑…

手机k歌用什么麦克风最好?轻揭无线麦克风哪个品牌音质最好!

​无线领夹麦克风作为现代音频技术的重要代表&#xff0c;已广泛应用于各种场合。它不仅能提升演讲者声音质量&#xff0c;还能增加演讲互动性和生动性。然而面对众多产品如何选择适合自己的设备成难题。本文将提供选购使用无线领夹麦克风的建议与推荐款式&#xff0c;帮助你轻…

一文带你全面详细了解安全运维

一、安全运维-网络 1、IP地址相关 IP地址属于网络层地址&#xff0c;用于标识网络中的节点设备。 IP地址由32bit构成&#xff0c;每8bit一组&#xff0c;共占用4个字节。 IP地址由两部分组成&#xff0c;网络位和主机位。 IP地址分类&#xff1a; 类别网络位子网掩码私有地…

tomcat常用配置详解和优化方法

常用配置详解 1 目录结构 /bin&#xff1a;脚本文件目录。 /common/lib&#xff1a;存放所有web项目都可以访问的公共jar包&#xff08;使用Common类加载器加载&#xff09;。 /conf&#xff1a;存放配置文件&#xff0c;最重要的是server.xml。 /logs&#xff1a;存放日志文件…

vue 基于antV 实现流程图编辑器代码

最近在做流程图功能开发&#xff0c;发现阿里antV 有对应的可视化引擎&#xff0c;于是自己做了一个简单vue 基于antV 实现流程图编辑器代码 部分代码如下&#xff1a; <template><div id"flowEditorContent"><header><h3>antv X6 流程编辑…

iptables(4)规则匹配条件

简介 前面我们已经介绍了iptables的基本原理,表、链,数据包处理流程。如何查询各种表的信息。还有基本的增、删、改、保存的基础操作。 经过前文介绍,我们已经能够熟练的管理规则了,但是我们只使用过一种匹配条件,就是将”源地址”作为匹配条件。那么这篇文章中,我们就来…

搞IT需不需要考个软考中级?

如果你是在事业单位、银行、国企等体制内工作&#xff0c;建议考虑参加软考。通过软考评职称后&#xff0c;可以获得加薪和晋升的机会&#xff0c;而且晋升时也会更看重你的职称等级。我就是通过软考评定了中级职称&#xff0c;薪水涨了500元。 评职称并不仅仅是拿到证书就行&…

Talk|北京大学张嘉曌:NaVid - 视觉语言导航大模型

本期为TechBeat人工智能社区第602期线上Talk。 北京时间6月20日(周四)20:00&#xff0c;北京大学博士生—张嘉曌的Talk已经准时在TechBeat人工智能社区开播&#xff01; 他与大家分享的主题是: “NaVid - 视觉语言导航大模型”&#xff0c;NaVid是首个专为视觉语言导航&#xf…

在华为服务器上编译C++工程的若干错误以及排查方法和解决方法记录

目录 1 报错 2 查找错误原因 2.1 方法一&#xff1a;ldd命令 2.2 方法二&#xff1a;警告信息里面 3 解决错误 3.1 libpng16.so.16 和 libbrotlidec.so.1 问题 3.2 libdevmmap.so 和 libslog.so库问题 3.3 剩余错误 3.3.1 libacllite.so错误解决 3.3.2 libtaclstream…

方舟云康亏损收窄:三年近10亿销售成本,平均付费及月活仍大幅承压

《港湾商业观察》施子夫 三度递表后&#xff0c;终于通过聆讯&#xff0c;方舟云康控股有限公司(以下简称&#xff0c;方舟云康)有望近期内挂牌港交所。方舟云康的国内运营主体为广州方舟云康信息科技集团有限公司、广州方舟医药有限公司。 值得关注的是&#xff0c;亏损的难…

【Hadoop大数据技术】——期末复习(冲刺篇)

&#x1f4d6; 前言&#xff1a;快考试了&#xff0c;做篇期末总结&#xff0c;都是重点与必考点。 题型&#xff1a;简答题、编程题&#xff08;Java与Shell操作&#xff09;、看图分析题。题目大概率会从课后习题、实验里出。 课本&#xff1a; 目录 &#x1f552; 1. HDF…

基于YOLOv5的火焰烟雾检测系统

基于YOLOv5的火焰烟雾检测系统 通过PYQT构建UI界面&#xff0c;包含图片检测&#xff0c;视频检测&#xff0c;摄像头实时检测。 &#xff08;该系统可以根据数据训练出的yolov5的权重文件&#xff0c;运用在其他检测系统上&#xff0c;如吸烟检测&#xff0c;口罩检测等等&…

常用快捷键-快速开发-mac idea 查看侧边栏tool window project+新建文件快捷键

背景&#xff1a;来到公司后&#xff0c;换了mac系统&#xff0c;有点不习惯&#xff0c;于是自己重新设置了开发的快捷键 1、mac idea 查看侧边侧栏&#xff08;专业说法是Tool Window Project&#xff09;&#xff1a; 每次我们都要点击一下左上角的这个类似于文件夹的图标…

简单高效的盈利策略,昂首资本推荐价格行为交易

有没有这样一种简单高效的盈利策略&#xff0c;不仅易于新手掌握&#xff0c;也是专业人士的常用利器?当然有了&#xff0c;就是Anzo Capital昂首资本今天推荐的价格行为交易。价格行为交易以其透明清晰的市场视角受到交易员的青睐&#xff0c;它如实反映了市场的真实动态&…

stm32学习笔记---STM32基础介绍

目录 STM32介绍 STM32家族系列 ARM介绍 ARM内核型号种类 我们学习用的STM32 片上资源/外设&#xff08;Peripheral&#xff09; 命名规则 系统结构 引脚定义 STM32的启动配置 STM32最小系统电路和其他部分电路 最小系统板的实物图 附&#xff1a;安装软件准备 声明…

netcore 生成验证码

安装依赖 Install-Package Lazy.Captcha.Core 注册服务 builder.Services.AddCaptcha(); 自定义注册服务 // 注册服务的时候增加配置 services.AddCaptcha(Configuration, option > {option.CaptchaType CaptchaType.WORD; // 验证码类型option.CodeLength 6; // 验证…

ARM32开发--FreeRTOS-事件组

系列文章目录 知不足而奋进 望远山而前行 目录 系列文章目录 文章目录 前言 目标 内容 概念 事件标志位 开发流程 功能介绍 创建事件组 触发事件 等待事件触发 同步 清理事件 案例 总结 前言 在嵌入式系统开发中&#xff0c;任务之间的同步和通信是至关重要的…