Linux文件编程详解

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;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值