练习所学文件操作的相关函数

简介: 上上篇文章,我们介绍了文件和文件操作函数,现在我们来练习一下所学文件操作的相关函数吧!实践出真知~

上上篇文章,我们介绍了文件和文件操作函数,现在我们来练习一下所学文件操作的相关函数吧!

实践出真知~

文件的打开和关闭

我们首先练习一下文件的打开和关闭:

每个被使⽤的⽂件都在内存中开辟了⼀个相应的⽂件信息区,⽤来存放⽂件的相关信息(如⽂件的名字,⽂件状态及⽂件当前的位置等)。这些信息是保存在⼀个结构体变量中的。该结构体类型是由系统声明的,取名FILE。

truct _iobuf {
  char *_ptr;
  int _cnt;
  char *_base;
  int _flag;
  int _file;
  int _charbuf;
  int _bufsiz;
  char *_tmpfname;
};
typedef struct _iobuf FILE;

⼀般都是通过⼀个FILE的指针来维护这个FILE结构的变量,这样使⽤起来更加⽅便。

下⾯我们可以创建⼀个FILE*的指针变量:

FILE* pf;//⽂件指针变量

定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变量)。通过该⽂件信息区中的信息就能够访问该⽂件。也就是说,通过⽂件指针变量能够间接找到与它关联的⽂件。

 

如图所示,我们在文件里写入成功了。

# define _CRT_SECURE_NO_WARNINGS 
#include <stdio.h>
int main()
{
  FILE* pFile;
  //打开⽂件
  pFile = fopen("project2", "w");
  //⽂件操作
  if (pFile != NULL)
  {
    fputs("fopen example", pFile);
    //关闭⽂件
    fclose(pFile);
  }
  return 0;
}

让我们打开这个文件,看看到底发生了什么吧。

接着让我们来到文件的随机读写

fseek

根据⽂件指针的位置和偏移量来定位⽂件指针。

int fseek ( FILE * stream, long int offset, int origin );


8935138c51d041de25166258c178322c_59ab6d95da4f4fbda9bd1d91760d4ffb.png

ftell

long int ftell ( FILE * stream );
#include <stdio.h>
int main ()
{
  FILE * pFile;
  long size;
  pFile = fopen ("myfile.txt","rb");
  if (pFile==NULL)
  perror ("Error opening file");
else
 {
  fseek (pFile, 0, SEEK_END); // non-portable
  size=ftell (pFile);
  fclose (pFile);
  printf ("Size of myfile.txt: %ld bytes.\n",size);
 }
  return 0;
}

rewind

void rewind ( FILE * stream );

让文件指针回到起始位置

#include <stdio.h>
int main ()
{
  int n;
  FILE * pFile;
  char buffer [27];
  pFile = fopen ("myfile.txt","w+");
  for ( n='A' ; n<='Z' ; n++)
    fputc ( n, pFile);
  rewind (pFile);
  fread (buffer,1,26,pFile);
  fclose (pFile);
  buffer[26]='\0';
  printf(buffer);
  return 0;
}

这段代码的主要功能是:


  1. 打开一个名为 "myfile.txt" 的文件,使用 "w+" 模式,这表示文件会被打开用于读写。如果文件不存在,它会被创建。如果文件已存在,它的内容会被清空。
  2. 使用 for 循环从字符 'A' 到 'Z',并将每个字符写入到文件中。
  3. 使用 rewind 函数将文件指针重新定位到文件的开头。
  4. 使用 fread 函数从文件中读取26个字节到 buffer 数组中。
  5. 关闭文件。
  6. 在 buffer 的第27个位置(索引为26)添加一个空字符(null terminator)\0,以确保 buffer 是一个合法的C字符串。
  7. 使用 printf 打印 buffer 的内容。

现在,让我们分析代码的结果:


  • 当循环执行时,它会将字符 'A' 到 'Z' 写入文件。
  • 使用 fread 读取时,会读取这26个字符。
  • 但是,这里有一个问题。fread 读取的字节不会转换为字符串的终止符,因此在添加 \0 之前,buffer 并不是一个合法的C字符串。但是,在这段代码中,你确实在读取后添加了 \0,所以这不是一个错误。

最终,buffer 将包含字符 'A' 到 'Z',并且以 \0 结尾。因此,当使用 printf 打印 buffer 时,它将输出:

ABCDEFGHIJKLMNOPQRSTUVWXYZ

文件结束的判定

文本文件:

#include <stdio.h>
#include <stdlib.h>
int main(void)
{
  int c; // 注意:int,⾮char,要求处理EOF
  FILE* fp = fopen("test.txt", "r");
  if(!fp) {
  perror("File opening failed");
  return EXIT_FAILURE;
}
//fgetc 当读取失败的时候或者遇到⽂件结束的时候,都会返回EOF
  while ((c = fgetc(fp)) != EOF) // 标准C I/O读取⽂件循环
  {
     putchar(c);
  }
//判断是什么原因结束的
  if (ferror(fp))
    puts("I/O error when reading");
  else if (feof(fp))
     puts("End of file reached successfully");
   fclose(fp);
}

二进制文件:

#include <stdio.h>
enum 
{ 
SIZE = 5 
};
int main(void)
{
  double a[SIZE] = {1.,2.,3.,4.,5.};
  FILE *fp = fopen("test.bin", "wb"); // 必须⽤⼆进制模式
  fwrite(a, sizeof *a, SIZE, fp); // 写 double 的数组
  fclose(fp);
  double b[SIZE];
  fp = fopen("test.bin","rb");
  size_t ret_code = fread(b, sizeof *b, SIZE, fp); // 读 double 的数组
  if(ret_code == SIZE) {
    puts("Array read successfully, contents: ");
    for(int n = 0; n < SIZE; ++n)
       printf("%f ", b[n]);
   putchar('\n');
 } 
   else { // error handling
     if (feof(fp))
       printf("Error reading test.bin: unexpected end of file\n");
     else if (ferror(fp)) {
       perror("Error reading test.bin");
        }
}
  fclose(fp);
}

文件缓冲区

#include <stdio.h>
#include <windows.h>
//VS2019 WIN11环境测试
int main()
{
  FILE*pf = fopen("test.txt", "w");
  fputs("abcdef", pf);//先将代码放在输出缓冲区
  printf("睡眠10秒-已经写数据了,打开test.txt⽂件,发现⽂件没有内容\n");
  Sleep(10000);
  printf("刷新缓冲区\n");
  fflush(pf);//刷新缓冲区时,才将输出缓冲区的数据写到⽂件(磁盘)
  //注:fflush 在⾼版本的VS上不能使⽤了
  printf("再睡眠10秒-此时,再次打开test.txt⽂件,⽂件有内容了\n");
  Sleep(10000);
  fclose(pf);
  //注:fclose在关闭⽂件的时候,也会刷新缓冲区
  pf = NULL;
  return 0;
}


相关文章
|
9月前
|
前端开发 JavaScript
Vue2下载二进制文件
Vue2下载二进制文件
130 6
|
API 数据安全/隐私保护 开发者
实时获取小红书详情 API 数据
小红书详情API数据获取指南:注册开发者账号,创建应用并申请接口权限,构建请求获取笔记详情,使用Python等语言处理响应数据。需遵守使用规则,注意调用频率和数据安全。
|
10月前
|
Cloud Native 安全 Java
开源之夏经验分享|Koupleless 社区黄兴抗:在开源中培养工程思维
黄兴抗是南昌师范学院电子信息工程专业的大三学生,同时也是Koupleless社区的贡献者。在开源之夏2024项目中,他参与了“存量应用自动改造成模块”的开发,旨在解决企业云原生转型中的存量应用改造难题。通过自动化工具,实现了传统应用向模块化的低成本升级,兼顾代码兼容性与独立启动功能。项目链接:[点击这里](https://summer-ospp.ac.cn/org/prodetail/2495a0376?lang=zh&list=pro)。 简介字数:238个字符。
|
设计模式 监控 Java
深入理解Spring Cloud中的断路器模式
深入理解Spring Cloud中的断路器模式
|
网络协议 Unix Linux
TCP 三次握手、四次断开
TCP 三次握手、四次断开
|
存储 人工智能 安全
尽管存在疑虑,但CIO仍在继续发掘和利用AIGC的优势
尽管存在疑虑,但CIO仍在继续发掘和利用AIGC的优势
|
JavaScript 前端开发 API
【Vue3学习笔记】Vue3入门学习
本笔记跟随Bilibili尚硅谷张天禹讲师的Vue全家桶课程学习,非常适合不了解Vue3的同学们入门观看!Vue2笔记共三篇 Vue3笔记一篇
1149 1
|
消息中间件 缓存 Java
Kafka介绍
Kafka是由Apache软件基金会开发的一个开源流处理平台,由Scala和Java编写。 Kafka是一种高吞吐量的分布式发布订阅消息系统,作为消息中间件来说都起到了系统间解耦、异步、削峰等作用,同时又提供了Kafka streaming插件包在应用端实现实时在线流处理,它可以收集并处理用户在网站中的所有动作流数据以及物联网设备的采样信息
317 0
|
存储 分布式计算 监控
分布式数据库HBase的基本概念和架构之基本架构的Region Server
分布式数据库HBase是一个开源的分布式数据库系统,是Apache Hadoop生态系统的重要组成部分。
560 0
|
Prometheus Cloud Native Go
Docker Buildx 构建多系统架构镜像
董江,容器技术布道者及实践者,中国移动高级系统架构专家,曾担任华为云核心网技术专家,CloudNative社区核心成员,KubeServiceStack社区发起者,Prometheus社区PMC,Knative Committer,Grafana社区Contributer。 欢迎关注:https://kubeservice.cn/