C/C++程序从文本文件中读取(保存)数据

:本文仅供初学者参阅,解惑

 

  • 在C程序中:

与程序代码外的数据(文件)打交道,我们使用到流(stream)这个概念,实现进程的虚拟内存与文件之间的数据交换。

——文件流:C标准库提供了FILE(之所以命名为FILE,因为linux将所有机制都视为文件) ,FILE对象是一个包含了管理流所需的所有信息的结构,包括缓冲区信息、各种标记(如文件结束标记和错误标记)以及用于实际I/O的文 件描述符等。

——输入流,输入流:数据从文件传送到内存的叫输入流,数据从内存传送到文件的叫输出流。

——打开文件:FILE对象通过调用fopen函数创建的。如: FILE *fp, fp=fopen("filename","r"), 表示以只读的方式建立与filename相关的文件流;filename为当前目录下的相对路径名,r代表可读(打开文件的模式)。 

 

一:读取

1:对一些有规范格式文件的读取,可使用标准库stdio.h下的fscanf函数,

函数原型为:int fscanf(FILE * stream, const char * format, [argument...])

如读取文件data.txt(数据格式相对规范)

代码实现读取:

//test.c
//文件读取
  #include<stdio.h>
  
  int main()
  {
    //1:创建文件流,文件指针名=fopen(文件名,使用文件方式)打开失败则返回NULL;
    FILE *fp=fopen("./data.txt","r"); //以data.txt文件为例

    //2:检测文件是否打开成功;
    if(!fp){
       printf("打开失败!\n");
       return -1; //返回异常
    }
    //3:
    int num; //用来储存一个整型数据
    char name[10], place[10]; //用来储存两个字符串数据

    //抽象理解:
    //理解文件位置含义:其表示已打开文件当前可读写字符的位置,其表示为一个到文件头的整数;
    //fscanf在读取数据时可以这样理解:文件被打开后,它就成了一个无序字节流(水流),其会通过一个管道,流向被读取的一端;
    //了解fscanf后知道,其遇到空格字符(空格,制表符),换行符,就会停止,这里的停止我们可以理解为:
    //停下来为文件中两个不相关的数据块做一个分隔的操作,刚好适应了我们一般将空字符(包括换行符)作为两个数据之间的分隔的行为;
    //我们只需理解管道的两端
    //流进管道的一端的位置,就是文件的位置,表示已被读到的位置。
    //流出管道一端,就是进程用来读取数据的一端,其可以对管道中已经做区分的数据进行读取。
   
   //4:读取:
    fscanf(fp,"%d%s%s",&num, name,place); //fscanf对流的格式化读取。
    //注1:fscanf(fp,"%da%s%s",&num,name,place); 可以实现对数据:1a小刚 河南;的准确读取,表示两个数据之间以a为界。
    //注2:因为流是指针的性质,所以函数是将各数据块的首地址交给对应参数,所以num需进行&取地址操作,
    //注3:因为name,place本身已表达地址,所以不用改变;

    fscanf(fp,"\n");
    //\n为控制字符,此时文件的位置到了第二行的开始;

    //接着进行操作:fscanf(fp,"%d%s%s",&num, name,place);就可以继续读取第二行
    //所以我们常常只需利用一个while语句就可以将整个文件读取到一个数据结构(进程)中
    /*
    while(!feof(fp)) //feof()检测一个文件是否结束,即到达文件尾,若结束,则返回非0值,否则返回0
    {
       fscanf(fp,"%d%s%s\n",&num, name,place); 
    }
    */
 
    //检测读取结果
    printf("%d %s %s\n",num, name, place);
 
    //关闭流
    fclose(fp); 
    
    return 0;

 }

程序运行结果:

 

2:从文件中读取整行数据(标准库stdio.h下的fgets

函数原型:char *fgets(char *str, int n, FILE *stream); 

其从指定的流 stream 读取一行,并把它存储在 str 所指向的字符串内。当读取 (n-1) 个字符时,或者读取到换行符时,或者到达文件末尾时,它会停止,遇到空格不停止;

例从文件中读取一行数据:

代码实现:

//test3.c
//行读取文件数据

#include<stdio.h>
#define maxlen 30

int main()
{
    //创建文件流
    FILE *fp=fopen("./data.txt","r");
    
    //2:检测文件是否打开成功;
    if(!fp){
       printf("打开失败!\n");
       return -1; //返回异常
    }
    char str[maxlen];//缓冲区,用来储存数据

    //从文件中读取一行数据,储存到str开始的地址,最大长度为maxlen,然后下次读取从下行开始
    //如果该行的数据长于maxlen-1,则只能返回一个不完整的行,并下次调用时从该处开始
    fgets(str,maxlen,fp);

    //检测结果
    printf("%s\n",str);

    //关闭流
    fclose(fp);
    return 0;

}

运行结果:

 


 

二:保存

1:保存与读取往往相关联,保存格式决定了你读取的方式,使用函数fprintf可以进行指定格式的保存:

函数原型为:int fprintf( FILE *stream, const char *format, [ argument ]...) 

假设保存一个人的个人信息到文件中:

/*test2.c */
//数据保存

#include<stdio.h>

int main()
{
    //例一个人的信息
    int num=1;
    char name[10]="小明";
    char place[10]="河南";

    //建立与文件的流
    FILE *fp=fopen("./data.txt","w");
    
    //2:检测文件是否打开成功;
    if(!fp){
       printf("打开失败!\n");
       return -1; //返回异常
    }

    //将数据格式化输出到指定文件流,int fprintf( FILE *stream, const char *format, [ argument ]...) 
    //注:此函数,是将format字符串写入到指定输出流中,format包括空格字符,非空格字符,说明符之中的一个或多个。如:fprintf(fp," "); 就是将空格输入到流中。
    //可理解为进程借助流将数据打印(fprintf)到了文件中;

    //将个人信息,写入指定流中,数据间以一个空格分隔,最后还写入换行符(控制字符)。
    fprintf(fp,"%d %s %s\n",num, name,place);

    //所以常常只要利用一个while语句就可以将讲一个表(链表,顺序表)按指定行格式写入输出流中
    /*
    while(!feof(fp)) //feof()检测一个文件是否结束,即到达文件尾,若结束,则返回非0值,否则返回0
    {
       fprintf(fp,"%d %s %s\n",num, name,place); 
    }
    */
    //关闭流
    fclose(fp);

    return 0;

}

运行结果:

 

2:写一个字符串到流中(fputs),函数原型:int fputs(const char *str, FILE *stream);

例:

//test4.c
//保存字符串

#include<stdio.h>

int main()
{
    //1:创建文件流,文件指针名=fopen(文件名,使用文件方式)打开失败则返回NULL;
    FILE *fp=fopen("./data.txt","a"); //以data.txt文件为例,a表示追加

    //2:检测文件是否打开成功;
    if(!fp){
       printf("打开失败!\n");
       return -1; //返回异常
    }

    //string
    char string[20]="Facing the world";

    //write string to the fstream
    fputs(string,fp);

    //关闭流
    fclose(fp);

    return 0;

}

运行结果:

 

 

浅谈c++:


 

在c++中我们可以使用操作符<<, >>来进行流的读写操作,更加的方便和易于理解;

具体参考下列实例:

1:读取所示数据:

代码实现:

//c++文件读取
#include<iostream> //输入输出流
#include<fstream> //文件流

//using namespace std; //若使用该声明,则可以不用在使用的每个标准库的成员前加std::


int main()
{
   //序号,年龄,年;
   int num, age, year;
   //姓名,地址
   char name[20], place[20];

   //c++的文件流,ifstream为输入文件流
   std::ifstream fp;

   //open为ifstream的成员函数,功能为打开文件,并将它与流关联
   fp.open("./data.txt",std::ios::in); //ios::in表示读流的方式,表示打开模式。

   //成员函数is_open检查流是否有关联文件,即打开成功与否,成功返回true,失败返回false
   if(!fp.is_open()){
       std::cout<<"打开文件失败!!\n";
       return 1;       // 返回异常;
   }

   //读取数据
   fp>>num>>year>>age>>name>>place; //使用操作符>>,将数据传输到对应的变量中

   //检测
   std::cout<<num<<":"<<name<<",age:"<<age<<",year:"<<year<<",live in:"<<place<<"\n"; //cout相当于printf

   //关闭流
   fp.close();

    return 0;
}

运行结果:

 

2:往文件保存数据:

实例:保存一个人的具体信息到文件data.txt中

代码实现:

//c++数据保存
#include<iostream> //输入输出流
#include<fstream> //文件流

//using namespace std; //若使用该声明,则可以不用在使用的每个标准库的成员前加std::


int main()
{
   //序号,年龄,年;
   int num=3;
   int age=20;
   int year=1993;
   //姓名,地址
   char name[20]="小龙";
   char place[20]="广元";

   //c++的文件流,ofstream为输出文件流
   std::ofstream fp;

   //open为ofstream的成员函数,功能为打开文件,并将它与流关联
   fp.open("./data.txt",std::ios::app); //ios::app表示每次写入是都追加到流尾,表示打开模式。

   //成员函数is_open检查流是否有关联文件,即打开成功与否,成功返回true,失败返回false
   if(!fp.is_open()){
       std::cout<<"打开文件失败!!\n";
       return 1;       // 返回异常;
   }

   //读取数据
   fp<<num<<" "<<year<<" "<<age<<" "<<name<<" "<<place<<"\n"; //使用操作符<<,将各数据传输到流所关联的文件中

   //关闭流
   fp.close();

    return 0;
}

运行结果:

 

posted on 2019-12-27 11:23  斤斤计较了  阅读(44640)  评论(15编辑  收藏  举报