【C/C++】详解不同场景下的文件读写操作

🔗 运行环境:C/C++

🚩 撰写作者:左手の明天

🥇 精选专栏:《python》

🔥  推荐专栏:《算法研究》

🔐#### 防伪水印——左手の明天 ####🔐

💗 大家好🤗🤗🤗,我是左手の明天!好久不见💗

💗今天分享C/C++——不同场景下的文件读写操作💗

📆  最近更新:2024 年 05 月 13 日,左手の明天的第 330 篇原创博客

📚 更新于专栏:C/C++入门与进阶

🔐#### 防伪水印——左手の明天 ####🔐


目录

1、文件模式

1.1 文本模式

1.2 二进制模式

2、文件读写操作函数

2.1 读写操作函数

2.2 文件使用模式

3、写入数据到文件

4、从文件中读取数据

5、在文件中移动读取位置

6、获取当前读取位置

7、文件读取每一行字符串赋值给不同的人

7.1 解决方案1:使用fgets()函数

7.2 解决方案2:使用fgetc()函数

7.3 解决方案3:使用fscanf()函数

7.4 解决方案4:使用 POSIX 的 getline() 函数

7.5 注意

8、文件读取每一行字符串赋值给结构体

8.1 解决方案1:使用fgets()和sscanf()

8.2 解决方案2:使用fgets()和strtok()

9、工程实例


1、文件模式

C把文件看作是一系列的连续的字节,每个字节都能被单独读取。

C提供两种文件模式:文本模式和二进制模式。首先要区分文本内容和二进制内容,文本文件格式和二进制文件格式,及文件的文本模式和二进制模式。

1.1 文本模式

文本内容:如果文件最初使用的是二进制编码的字符(如ASCII码)表示文本,该文件就是文本文件,其中包含文本内容。

1.2 二进制模式

二进制内容:如果文件中的二进制值代表机器语言代码或数值数据,图片或者音乐编码,该文件就是二进制文件,其中包含二进制内容。

2、文件读写操作函数

2.1 读写操作函数

在C语言中,文件的读写操作主要是通过以下几个函数来实现的:

  1. fopen():打开一个文件

  2. fclose():关闭一个文件

  3. fread():读取文件中的数据

  4. fwrite():写入数据到文件中

  5. fseek():设置文件的读取位置

  6. ftell():获取文件的当前读取位置 

2.2 文件使用模式

文件使用方式       含义                                        如果指定文件不存在
" r " (只读)      为了输入数据,打开一个已经存在的文本文件        出错
" w " (只写)      为了输出数据,打开一个文本文件                 建立一个新的文件
" a " (追加)      向文本文件尾部添加数据                        建立一个新的文件
" rb "  (只读)    为了输入数据,打开一个二进制文件               出错
" wb "  (只写)    为了输出数据,打开一个二进制文件               建立一个新的文件
" ab "  (追加)    向一个二进制文件尾部添加数据                  出错
" r+ "  (读写)    为了读写,打开一个文本文件                    出错
" w+ "  (读写)    为了读写,建立一个新的文本文件                 建立一个新的文件
" a+ "  (读写)    打开一个文件,在文件尾部进行读写               建立一个新的文件
" rb+ " (读写)    打开一个二进制文件(读写)                     出错
" wb+ " (读写)    建立一个新的二进制文件(读写)                 建立一个新的文件
" ab+ " (读写)    打开一个二进制文件,在文件尾部进行读和写        建立一个新的文件

3、写入数据到文件

#include <stdio.h>int main() {FILE *fp = fopen("test.txt", "w"); // 以写入模式打开文件if(fp == NULL) {printf("Failed to open file\n");return -1;}char str[] = "Hello, World!";fwrite(str, sizeof(char), sizeof(str), fp); // 将字符串写入文件fclose(fp); // 关闭文件return 0;}

4、从文件中读取数据

#include <stdio.h>int main() {FILE *fp = fopen("test.txt", "r"); // 以读取模式打开文件if(fp == NULL) {printf("Failed to open file\n");return -1;}char str[20];fread(str, sizeof(char), sizeof(str), fp); // 读取文件内容printf("%s", str); // 打印读取的内容fclose(fp); // 关闭文件return 0;
}

5、在文件中移动读取位置

#include <stdio.h>int main() {FILE *fp = fopen("test.txt", "r"); // 以读取模式打开文件if(fp == NULL) {printf("Failed to open file\n");return -1;}char str[20];fread(str, sizeof(char), sizeof(str), fp); // 读取文件的前部分内容fseek(fp, 0, SEEK_SET); // 将读取位置移动到文件开头fread(str, sizeof(char), sizeof(str), fp); // 再次读取文件的全部内容printf("%s", str); // 打印读取的内容fclose(fp); // 关闭文件return 0;
}

6、获取当前读取位置

#include <stdio.h>int main() {FILE *fp = fopen("test.txt", "r"); // 以读取模式打开文件if(fp == NULL) {printf("Failed to open file\n");return -1;}char str[20];fread(str, sizeof(char), sizeof(str), fp); // 读取文件内容long position = ftell(fp); // 获取当前读取位置printf("Current file position: %ld\n", position);fclose(fp); // 关闭文件return 0;
}

以上代码提供了文件的基本写入和读取操作,并展示了如何在文件中移动读取位置以及如何获取当前的读取位置。在实际应用中,可能需要添加错误检查和处理,以确保文件操作的正确性和稳健性。

7、文件读取每一行字符串赋值给不同的人

在C语言中,可以使用各种方法来读取文件中的每一行并将其赋值给不同的人。以下是一些可能的解决方案:

7.1 解决方案1:使用fgets()函数

#include <stdio.h>
#include <stdlib.h>#define MAX_NAME_LENGTH 100int main() {FILE *file = fopen("names.txt", "r"); // 打开包含名字的文件if (file == NULL) {perror("Error opening file");return EXIT_FAILURE;}char line[MAX_NAME_LENGTH];int person = 1; // 假设我们从第一个人开始while (fgets(line, sizeof(line), file) != NULL) {// 移除换行符size_t len = strlen(line);if (len > 0 && line[len-1] == '\n') {line[len-1] = '\0';}// 根据person变量的值,将名字分配给不同的人// 这里只是打印出来,实际应用中可能需要其他逻辑printf("将名字 '%s' 分配给人#%d\n", line, person);// 更新person的值person++;}fclose(file); // 关闭文件return EXIT_SUCCESS;
}

这个代码假设有一个文件names.txt,里面包含了一行行的名字。程序会逐行读取这些名字,并将它们输出,就好像它们被分配给了不同的人。可以根据你的具体需求来修改分配逻辑。

7.2 解决方案2:使用fgetc()函数

#include <stdio.h>
#include <stdlib.h>int main() {FILE *file = fopen("file.txt", "r");if (file == NULL) {printf("Failed to open file\n");return 1;}int ch;while ((ch = fgetc(file)) != EOF) {printf("%c", (char)ch);}fclose(file);return 0;
}

7.3 解决方案3:使用fscanf()函数

#include <stdio.h>
#include <stdlib.h>int main() {FILE *file = fopen("file.txt", "r");if (file == NULL) {printf("Failed to open file\n");return 1;}char buffer[1024];while (fscanf(file, "%s", buffer) != EOF) {printf("%s\n", buffer);}fclose(file);return 0;
}

7.4 解决方案4:使用 POSIX 的 getline() 函数

#include <stdio.h>
#include <stdlib.h>int main() {FILE *file = fopen("file.txt", "r");if (file == NULL) {printf("Failed to open file\n");return 1;}char *line = NULL;size_t len = 0;ssize_t read;while ((read = getline(&line, &len, file)) != -1) {printf("Retrieved line of length %zu:\n", read);printf("%s", line);   }fclose(file);if (line)free(line);return 0;}

7.5 注意

在所有这些示例中,都假设文件名为"file.txt",并且文件存在于程序可以访问的同一目录中。如果文件位于不同的目录或者你想要指定不同的文件名,你需要更改"file.txt"为你的文件路径。另外,所有这些示例都假设文件中的每一行都不会超过1024个字符(包括null终止符)。如果你的文件中的行可能会非常长,你需要使用动态内存分配来读取这些行。

8、文件读取每一行字符串赋值给结构体

在C语言中,可以使用fgets()函数从文件中逐行读取字符串,然后使用sscanf()函数或者strtok()函数来解析每一行的内容,并将其存储到结构体中。

8.1 解决方案1:使用fgets()和sscanf()

#include <stdio.h>
#include <stdlib.h>typedef struct {char name[256];int age;
} Person;int main() {FILE *file = fopen("data.txt", "r");if (file == NULL) {printf("Failed to open file\n");return 1;}Person person;char line[256];while (fgets(line, sizeof(line), file)) {sscanf(line, "%s %d", person.name, &person.age);printf("Name: %s, Age: %d\n", person.name, person.age);}fclose(file);return 0;}

8.2 解决方案2:使用fgets()和strtok()

#include <stdio.h>
#include <string.h>
#include <stdlib.h>typedef struct {char name[256];int age;
} Person;int main() {FILE *file = fopen("data.txt", "r");if (file == NULL) {printf("Failed to open file\n");return 1;}Person person;char line[256];char *token;while (fgets(line, sizeof(line), file)) {token = strtok(line, " ");strcpy(person.name, token);token = strtok(NULL, " ");person.age = atoi(token);printf("Name: %s, Age: %d\n", person.name, person.age);}fclose(file);return 0;}
在这两个例子中,我们都假设文件"data.txt"的内容是这样的:

John Doe 30

Jane Smith 25

每一行包括姓名和年龄,并用空格分开。这两个例子都将文件中的每一行读取出来,然后将其赋值给Person结构体。

注意:这两个例子都没有进行错误检查,在实际使用时应该添加错误检查。例如,在调用fopen()、fgets()、sscanf()、strtok()以及atoi()时都可能出错,并且在出错时应该有恰当的处理方式。

9、工程实例

从键盘输入若干行字符,输入后把他们存储到一磁盘文件中。在从该文件中读出这些数据,将其中的小写字母转换成大写字母后在屏幕上输出。

#include <stdio.h>
#include<stdlib.h>struct node
{int data;struct node *next;
} ;typedef  struct  node  LNode;  
typedef  struct  node * LinkList;  
void CreateHeadList (LinkList  L);   
void OutputList(LinkList L);       
void DestroyList(LinkList  L);  void main()
{LinkList  head;          //定义一个LinkList 型的变量headif( ( head =(LNode *) malloc( sizeof(LNode) ) )==NULL)  {  printf("申请空间失败!");  exit(0);  }  head ->next=NULL;      //新建了一个空的单链表headCreateHeadList ( head);     //用头插法输入数据创建各结点OutputList(head);         //输出以head为头的链表各结点的值DestroyList (head);        //销毁以head为头的链表各结点
}/*定义头插法创建带头结点的单链表的函数*/
void CreateHeadList(LinkList  L)  
{	LNode *p;int  x, flag=-1;  scanf("%d",&x);while (x != flag){  if( ( p =(LNode *) malloc( sizeof(LNode) ) )==NULL)  {  printf("申请空间失败!");  exit(0);  }  p->data=x;      //x的数据类型为int型p->next=L->next;  L->next=p;  scanf("%d",&x);}  
} /*定义输出带头结点的单链表的函数*/
void OutputList(LinkList  head)  
{struct node *p;p=head->next ;           //取得链表的头指针while(p!=NULL)    //只要p指向的结点非空{printf("%d\t",p->data);       //p->data数据类型为int型p=p->next;                 //p指向该结点的下一个结点}printf("\n");
} /*定义销毁带头结点的单链表的函数*/
void  DestroyList (LinkList  head) 
{struct node  *p, *q;p=head;              //取得链表的头指针while(p!=NULL)       //只要p指向的结点非空{q=p->next;        //将p的下一个结点的地址保存在q中free(p);           //释放p指向的结点p=q;              }
}

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

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

相关文章

虚拟数字人及AI相关应用分享

一、虚拟数字人 1、简介 虚拟数字人可分为基础类和仿真智能类。可用于直播的&#xff0c;一般是仿真智能类&#xff1b;基础类动作缓慢&#xff0c;体验差&#xff0c;很容易被直播平台封号。 目前各大短视频平台上介绍的数字人&#xff0c;出于营销目的&#xff0c;有夸大宣传…

代码随想录35期Day38-Java(Day37休息)

Day38题目 LeetCode509.斐波那契数列 核心思想:很简单dp[i]dp[i-1]dp[i-2].这里用了数组存储的形式,也可以递归 class Solution {public int fib(int n) {int[] dp new int[n2];dp[0] 0;dp[1] 1;for(int i 2 ; i < n ; i ){dp[i] dp[i-1] dp[i-2];}return dp[n];} …

go语言冒泡排序算法之泛型实现 支持整型,浮点型和字符串排序 golang bubble sort general

冒泡排序是一个比较经典的基础算法&#xff0c; 是学习编程的同学们所必备的算法之一&#xff0c;今天就介绍一个go语言里面使用泛型实现的冒泡排序算法&#xff0c; 支持字符串 string&#xff0c;数字 int float 等的排序。 在go语言中通常我们在进行数据比较时都是使用的c…

prophet时间序列模型水质预测应用

前言 此前已经分析了&#xff0c;ARIMA 模型在水质预测中的应用&#xff0c;今天用 prophet 模型测试下在水质预测中的效果。 Prophet 简介 Prophet 是 Facebook 于2017年开源的一个时间序列预测框架&#xff0c;特别适合于处理具有明显趋势性和季节性的数据。该模型设计初衷…

领略Java内部类的“内部”

内部类有两种情况&#xff1a; (1) 在类中定义一个类(私有内部类&#xff0c;静态内部类) (2) 在方法中定义一个类(局部内部类&#xff0c;匿名内部类) 1、私有内部类 —— 在方法之间定义的内部类&#xff0c;非静态 我们首先看看类中内部类的两个特点&#xff1a; (1) 在外部…

springboot3项目练习详细步骤(第三部分:文章管理模块)

目录 发布文章 接口文档 业务实现 自定义参数校验 项目参数要求 实现思路 实现步骤 文章列表(条件分页) 接口文档 业务实现 mapper映射 更新文章 接口文档 业务实现 获取文章详情 接口文档 业务实现 删除文章 接口文档 业务实现 文章管理业务表结构…

数据可视化训练第6天(美国人口调查获得关于收入与教育背景的数据,并且可视化)

数据来源 https://archive.ics.uci.edu/dataset/2/adult 过程 首先&#xff1b;关于教育背景的部分翻译有问题。 本次使用字典嵌套记录数据&#xff0c;并且通过lambda在sorted内部进行对某个字典的排序&#xff0c;最后用plotly进行绘图 本次提取数据的时候&#xff0c;用到…

ubuntu server 22.04.4 系统安装详细教程

本教程使用vmware workstation 17创建虚拟机进行安装演示&#xff0c;安装方式和真机安装没有区别。 1、下载镜像 下载ubuntu server版本系统镜像&#xff0c;官网下载地址&#xff1a;https://cn.ubuntu.com/download/server/step1 注意&#xff1a;自己下载时需要确认是否是…

MySQL索引的存储结构

2.3.2. 索引存储结构 索引存储结构有堆表和索引组织表两种方式。 堆表和索引组织表有什么区别&#xff1f; 堆表&#xff1a; 数据和索引是分开存放的&#xff0c;索引是排序后的&#xff0c;但堆数据是无序的进行的都是随机访问&#xff0c;索引的叶子节点中存放的是数据在…

基于springboot + vue 实现的简易博客系统

项目效果图 登陆页面 文章列表 发表文章 用户管理 栏目管理 数据统计 后端技术栈后端主要采用了&#xff1a; 1.SpringBoot 2.SpringSecurity 3.MyBatis 4.部分接口遵循Restful风格 5.MySQL 前端技术栈前端主要采用了&#xff1a; 1.Vue 2.axios 3.Elemen…

每周一算法:传递闭包

题目描述 不等式排序 给定 n n n个变量和 m m m个不等式。其中 n n n小于等于 26 26 26&#xff0c;变量分别用前 n n n 的大写英文字母表示。 不等式之间具有传递性&#xff0c;即若 A > B A>B A>B 且 B > C B>C B>C&#xff0c;则 A > C A>C …

GPU prompt

提问&#xff1a; GPU是如何与CPU协调工作的&#xff1f; GPU也有缓存机制吗&#xff1f;有几层&#xff1f;速度差异是多少&#xff1f; GPU渲染流程有哪些阶段&#xff1f;他们的功能分别是什么&#xff1f; Early-Z技术是什么&#xff1f;发生在哪个阶段&#xff1f;这个…

Cocos 2048从创建到发布上线

二、审核通过之后上线流程 代码通过审核之后&#xff0c;会通过站内信和微信消息发送通知&#xff0c;在管理后台&#xff0c;点击提交发布&#xff0c;去备案

ensp简单ac+ap+sta无线配置和脚本

接入交换机与ap连线配置: interface E0/0/5 port link-type trunk port trunk pvid vlan 10 port trunk allow-pass vlan all配置AC的IP配置&#xff1a; [AC]Vlan 2 创建vlan 2 [AC]interface Vlanif 2 进入vlan 2 [AC-Vlanif2]ip address 192.168.80.3 24 设置一个IP地址…

File类~路径、创建文件对象

路径分为相对路径&#xff08;不带盘符&#xff09;&#xff0c;绝对路径&#xff08;带盘符&#xff09; 路径是可以存在的&#xff0c;也可以是不存在的 创建文件对象的三个方法&#xff1a;

IDEA-SpringBoot项目启动类位置错误导致启动报错

文章目录 前言&#xff1a;本文为SpringBoot项目启动类位置错误导致启动报错解决方案 笔者在初学SpringBoot项目时遇到过因启动类位置错误导致项目启动报错现象&#xff0c;总结如下&#xff1a; SpringBoot启动发送请求后遇到如下错误&#xff1a; Postman界面信息&#xff1…

速了解及使用布隆过滤器

布隆过滤器 介绍 概念&#xff1a;是一种高效查询的数据结构 作用&#xff1a;判断某个元素是否在一个集合中。&#xff08;但是会出现误判的情况&#xff09; 实现原理 加入元素&#xff1a; 当一个元素需要加入到布隆过滤器中时&#xff0c;会使用一组哈希函数对该元素进…

个人模拟面试java2

文章目录 SQL如何进行优化数据库性能调优1. 性能评估2. 确定调优目标 3. 监控和分析4. 硬件和配置调优5. 数据库设计和模式优化6. 查询优化7. 并发控制8. 缓存策略9. 定期维护10. 监控和持续调优 Redis支持哪些数据类型&#xff1f;Redis如何实现持久化&#xff1f;如何解决Red…

☀️提高生产力:这10个Lambda表达式必须掌握!!!

1.使用Lambda表达式进行集合遍历 ❌ 未使用Lambda表达式: List list Arrays.aslist("apple", "banana", "orange"); for (String fruit : list) {System.out.println(fruit); }✅使用Lambda表达式: List list Arrays.aslist("apple&qu…

Dubbo为何放弃JDK SPI机制:一探究竟

引言 在当今的软件开发领域&#xff0c;随着微服务架构的普及和服务化治理的需要&#xff0c;Dubbo作为一款优秀的分布式服务框架&#xff0c;扮演着至关重要的角色。Dubbo以其强大的功能和易用性成为了众多企业和开发者的首选。然而&#xff0c;在Dubbo的设计中&#xff0c;我…