抽丝剥茧C语言(高阶)文件操作+练习(下)

简介: 抽丝剥茧C语言(高阶)文件操作+练习

5. 文件的随机读写

我们文件指针最开始指向的都是第一个数据,读完第一个又指向第二个,那么怎么来控制指向呢?

5.1 fseek

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

int fseek ( FILE * stream, long int offset, int origin );//第一个参数是流,第二个参数是偏移量,第三个参数是文件的起始位置

第一个是文件起始位置,第二个是当前文件指针的位置,第三个是文件的末尾位置。

例子:

#include <stdio.h>
int main()
{
  FILE* pFile;
  pFile = fopen("example.txt", "rb");
  //fputs("This is an apple.", pFile);
  fseek(pFile, 9, SEEK_SET);
  int i = fgetc(pFile);
  printf("%c\n", i);
  fseek(pFile, 2, SEEK_CUR);
  i = fgetc(pFile);
  printf("%c\n", i);
  fseek(pFile, -2, SEEK_END);
  i = fgetc(pFile);
  printf("%c\n", i);
  fclose(pFile);
  pFile = NULL;
  return 0;
}

这是文件里面的内容。

不要忘记字符串的末尾有个’\0‘。

这是结果。

5.2 ftell

返回文件指针相对于起始位置的偏移量

long int ftell ( FILE * stream );//参数是流

例子:

#include <stdio.h>
int main()
{
  FILE* pFile;
  pFile = fopen("example.txt", "rb");
  //fputs("This is an apple.", pFile);
  fseek(pFile, 9, SEEK_SET);
  int i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));//打印偏移量的位置
  fseek(pFile, 2, SEEK_CUR);
  i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  fseek(pFile, -2, SEEK_END);
  i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  fclose(pFile);
  pFile = NULL;
  return 0;
}

5.3 rewind

让文件指针的位置回到文件的起始位置

void rewind ( FILE * stream );

例子:

#include <stdio.h>
int main()
{
  FILE* pFile;
  pFile = fopen("example.txt", "rb");
  //fputs("This is an apple.", pFile);
  fseek(pFile, 9, SEEK_SET);
  int i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  fseek(pFile, 2, SEEK_CUR);
  i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  fseek(pFile, -2, SEEK_END);
  i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  rewind(pFile);//让指针回到起始位置
  i = fgetc(pFile);
  printf("%c\n", i);
  printf("%d\n", ftell(pFile));
  fclose(pFile);
  pFile = NULL;
  return 0;
}

回到了起始位置。

6. 文本文件和二进制文件

根据数据的组织形式,数据文件被称为文本文件或者二进制文件

数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件

如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件

一个数据在内存中是怎么存储的呢?

字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。

如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而二进制形式输出,则在磁盘上只占4个字节(VS2013测试)。

不考虑大小端的问题,下面是图解

测试代码

#include <stdio.h>
int main()
{
  int a = 10000;
  FILE* pf = fopen("test.txt", "wb");
  fwrite(&a, 4, 1, pf);//二进制的形式写到文件中
  fclose(pf);
  pf = NULL;
  return 0;
}

这里我的机器是小端存储的方式。

前面的00000000是编辑器的地址展示,不用管他。

7. 文件读取结束的判定

7.1 被错误使用的feof

牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。

而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束

  1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
    例如:
    fgetc 判断是否为 EOF .
    fgets 判断返回值是否为 NULL .
  2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
    例如:
    fread判断返回值是否小于实际要读的个数。//因为如果一共有9个字符,一次读5个,首先读了5个,那么再读一次剩下的4个字符返回值就是4,所以是小于。

正确的使用:

文本文件的例子:

#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>
#include <stdlib.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) //说明读到了5个元素
  {
    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);
}

文本文件的例子

8. 文件缓冲区

ANSIC 标准采用“缓冲文件系统”处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。

测试代码:

#include <stdio.h>
#include <windows.h>
//VS2013 WIN10环境测试
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;
}

刷新缓冲区会将缓冲区的内容都存到硬盘里。

这里可以得出一个结论:

因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文件。

如果不做,可能导致读写文件的问题。

本篇结束

相关文章
|
3月前
|
存储 编译器 程序员
c语言的文件操作与文件缓冲区
如果没有文件,我们写的程序的数据是存储在电脑的内存中,如果程序退出,内存回收,数据就丢失了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们可以使用文件。磁盘(硬盘)上的文件是文件。但是在程序设计中,我们⼀般谈的⽂件有两种:程序文件、数据文件(从文件功能的角度来分类 的)。就比如说我们电脑中以.txt为后缀的就是文件的一种,他就是数据文件。.exe为后缀的就为程序文件。函数名功能适用范围fgetc字符输入函数所有输入流fputc字符输出函数所有输出流fgets。
93 0
|
8月前
|
存储 小程序 C语言
【C语言程序设计——文件】文件操作(头歌实践教学平台习题)【合集】
本文介绍了C语言中的文件操作,分为两个关卡。第1关任务是将键盘输入的字符(以#结束)存入`file1.txt`并显示输出;第2关任务是从键盘输入若干行文本(每行不超过80个字符,用-1作为结束标志),写入`file2.txt`后再读取并显示。文中详细讲解了文件的打开、读取(使用`fgetc()`和`fgets()`)、写入(使用`fputc()`和`fputs()`)及关闭操作,并提供了示例代码和测试说明。
228 5
|
9月前
|
存储 程序员 C语言
【C语言】文件操作函数详解
C语言提供了一组标准库函数来处理文件操作,这些函数定义在 `<stdio.h>` 头文件中。文件操作包括文件的打开、读写、关闭以及文件属性的查询等。以下是常用文件操作函数的详细讲解,包括函数原型、参数说明、返回值说明、示例代码和表格汇总。
243 9
|
9月前
|
存储 数据管理 C语言
C 语言中的文件操作:数据持久化的关键桥梁
C语言中的文件操作是实现数据持久化的重要手段,通过 fopen、fclose、fread、fwrite 等函数,可以实现对文件的创建、读写和关闭,构建程序与外部数据存储之间的桥梁。
|
10月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
582 3
|
10月前
|
存储 C语言
【c语言】玩转文件操作
本文介绍了C语言中文件操作的基础知识,包括文件的打开和关闭、文件的顺序读写、文件的随机读写以及文件读取结束的判定。详细讲解了`fopen`、`fclose`、`fseek`、`ftell`、`rewind`等函数的使用方法,并通过示例代码展示了如何进行文件的读写操作。最后,还介绍了如何判断文件读取结束的原因,帮助读者更好地理解和应用文件操作技术。
202 2
|
11月前
|
存储 C语言
C语言文件操作(2)
【10月更文挑战第2天】
72 6
|
11月前
|
程序员 编译器 C语言
C语言底层知识------文件操作
本文详细介绍了文件操作的基本概念,包括文件的分类(程序文件和数据文件,其中着重于数据文件的文本文件和二进制文件),流的概念及其在C程序中的应用,以及标准输入输出流stdin、stdout和stderr的作用。作者通过示例展示了如何使用fopen、fclose和常见的读写函数如fgetc、fputc和fgets进行文件操作。
123 2
|
11月前
|
存储 C语言
简述C语言文件操作
简述C语言文件操作
62 0
|
11月前
|
存储 文件存储 C语言
深入C语言:文件操作实现局外影响程序
深入C语言:文件操作实现局外影响程序