【C语言】针对文件操作-打开-关闭-读写

文件的定义

首先我们需要了解,为什么需要使用文件,例如我们手写了一个通讯录代码,但是我们会发现,每次通讯录程序打开时,都需要将通讯录信息重新输入一遍,这显然是不合适的。那么此时我们就需要使用文件去存储我们想要保留的通讯录信息,以便再次打开时不用重新输入。通讯录代码我已经写好放到这里,大家批评指正。链接: 通讯录代码

我们需要对文件有一个初步的认识,当我们想把信息记录下来,就涉及到了数据持久化,简单来说就是数据要保存。我们可以选择将数据放在磁盘或者数据库。而我们使用文件就是将数据放在了我们电脑的磁盘上。

站在程序员的角度,我们一般会将文件分为两种:程序文件和数据文件。程序文件就是我们平时所见到的c文件,.exe文件 .py文件等等

而数据文件是当程序运行起来的时候产生的输出的文件。

当我们需要存储数据的时候,就需要打开文件,写入数据,

文件的打开和关闭

//使用fopen打开文件,fclose关闭文件
FILE *fp;//创建一个文件指针
fp = fopen("文件名","打开方式");
//当文件操作完成后要记得关闭文件
fclose(fp);

文件的打开方式有多种
在这里插入图片描述
请注意,当选择r作为打开方式时,若没有该文件,则会打开失败,而选择w作为打开方式,即使没有该文件,程序也会自动创建一个文件出来。
而我们应当对打开文件是否成功进行判断,这是一个良好的习惯

//使用fopen打开文件,fclose关闭文件
FILE *fp = NULL;//创建一个文件指针
fp = fopen("文件名","打开方式");
if(fp == NULL)//当文件打开失败时,fp指针为NULL
{
	printf("打开文件失败\n");
	return ;
}
//到这里说明打开成功

文件的读写

平时我们会将程序运行的结果显示在控制台,如果我们要对文件内容进行读写,那么也可以分为顺序读写和随机读写。

文件的顺序读写

在这里插入图片描述

文件的随机读写

fseek、 ftell、rewind都是针对文件指针进行操作的。

//fseek根据文件指针的位置和偏移量来定位文件指针。
int fseek ( FILE * stream, long int offset, int origin );
int main ()
{
  FILE * pFile;
  pFile = fopen ( "example.txt" , "wb" );
  fputs ( "This is an apple." , pFile );
  fseek ( pFile , 9 , SEEK_SET );
  fputs ( " sam" , pFile );
  fclose ( pFile );
  return 0;
}

在这里插入图片描述

//ftell返回文件指针相对于起始位置的偏移量
long int ftell ( FILE * stream );
int main ()
{
  FILE * pFile;
  long size;
  pFile = fopen ("myfile.txt","rb");
  if (pFile==NULL) perror ("Error opening file");
  else
 {
    fseek (pFile, 0, SEEK_END);
    size=ftell (pFile);
    fclose (pFile);
    printf ("Size of myfile.txt: %ld bytes.\n",size);
 }
  return 0;
}
 //rewind让文件指针的位置回到文件的起始位置
 void rewind ( FILE * stream );
 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';
  puts (buffer);
  return 0;
}

这里简单介绍一下二进制文件和文本文件,众所周知,我们的计算机都是以二进制存储信息的,那么当数据在内存中以二进制的方式存储,当不加任何转换输出到磁盘,那这就是二进制文件,如果我们以ASCII码存储到磁盘,而读取也是以ASCII码进行读取,那么这就是文本文件。

文件读取结束的判断

当我们需要对文件进行读取的时候,计算机并不知道文件到哪个位置结束,就需要我们自己去利用函数的返回值去判断文件是否读取完成。

对于一个文本文件,我们需要用fgetc(返回值为EOF)判断是否读取完成,fgets(返回值为NULL)判断是否读取完成。

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

对于一个二进制文件,我们需要用fread的返回值判断是否小于实际要读的个数判断是否结束。

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");
       }
   }
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值