C/C++:文本文件和二进制文件的读写

文本文件和二进制文件在计算机中的存储都是以0,1存储的,所不同的就是声明方式;

在C语言中,对于文本文件和二进制文件的读写,应该如何进行呢?

如何把数据保存为.txt文件?

如何读取.txt文件中的数据?

本经验主要包括以下几部分:

1、C++基础:数据流,缓冲区,文件类型

2、C语言:缓冲区文件处理:

3、文件读写流程:

4、文本文件操作:

5、格式化存取函数

6、二进制文件操作

C/C++:文本文件和二进制文件的读写

工具/原料

  • Visual Studio 2013
  • C++

方法/步骤

  1. 1、C++基础:数据流,缓冲区,文件类型

    ①数据流Stream:

    程序与数据间的交互是以流的形式进行的。

    在C语言文件存取时,

    都会先进行“打开文件”操作,目的是打开数据流;

    而“关闭文件”操作就是关闭数据流。

    ②缓冲区Buffer:

    程序执行时,提供的额外内存以暂时存放数据。

    缓冲区作用:为了提高存取效率,因为内存的存取速度比磁盘快。

    ③文件类型:

    分为文本文件和二进制文件两种。

    文本文件:字符编码的方式进行保存;

    二进制文件:内存中的数据原封不动至文件中,适用于非字符为主的数据;如果以记事本打开,只会看到一堆乱码。

    二进制文件优点:存取速度快,占用空间小,随时存取数据。

    C/C++:文本文件和二进制文件的读写
  2. 2、C语言:缓冲区文件处理:

    C语言文件处理功能根据系统是否设置“缓冲区”分为两种:

    一种是设置缓冲区;

    另一种是不设置缓冲区。

    由于不设置缓冲区的文件处理方式,必须使用较低级I/O函数来直接对磁盘读取,这种方式慢,并且由于不是C的标准函数,跨平台操作时容易出问题。

    本经验只介绍带缓冲区的文件处理方式:

    当使用在头文件stdio.h中的标准I/O函数时,系统会自动设置缓冲区,并通过数据流来读写文件。

    C/C++:文本文件和二进制文件的读写
  3. 3、文件读写流程:

    文件数据读取时,先打开数据流,将磁盘上的文件信息拷贝到缓冲区内,然后再从缓冲区中读取所需数据;

    当数据写入文件时,先将数据写入缓冲区,只有在缓冲区已满或“关闭文件”后,才会将数据写入磁盘;

    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
  4. 4、文本文件操作:

    C中主要通过标准I/O函数来对文本文件进行处理。

    文本文件操作包括:fopen(),fclose(),fputc(),fgets(),fputs(),fprintf(),fscanf()…

    打开文件函数fopen():

    函数原型:_CRTIMP FILE* __cdecl fopen(const char*,const char*);

    函数参数:第一个参数为文件名,第二个参数为打开模式。

    返回值:打开成功,fopen返回一个结构指针地址;否则返回NULL。

    示例:

    FILE *fp;

    fp=fopen(“c:\\temp\\test.txt”,”r”);//由于反斜杠\在C语言中是控制字符,所以为了区分再加一个反斜杠以表示路径。

    【注】:使用fopen()函数打开的文件会先将文件复制到缓冲区;在读取和写入操作中,都是针对缓冲区进行存取而不是磁盘,只有当fclose()函数关闭文件时,缓冲区中的数据才会写入磁盘。

    C/C++:文本文件和二进制文件的读写
  5. 4.1关闭文件

    函数原型:_CRTIMP int __cdecl fclose(FILE *);

    返回值:关闭成功返回值0,否则返回非零值。

    【注】:在执行完文件的操作后,要进行“关闭文件“操作。

    示例:打开文件和关闭文件

    C/C++:文本文件和二进制文件的读写
  6. 4.2字符存取函数fputc()/fgetc()

    函数原型:_CRTIMP int __cdecl fputc(int, FILE *);

              _CRTIMP int __cdecl fgetc(FILE *);

    fgetc()函数:字符读取函数,从文件数据流中一次读取一个字符,然后读取光标移动到下一个字符,并逐步将文件的内容读出。

    如果字符读取成功,则返回所读取的字符,否则返回EOF(end of file)。

    EOF是表示数据结尾的常量,真值是-1。

    判断文件是否读取完毕,可利用feof()函数进行检查。未读取结束返回0,已读取结束返回非零值。

    feof()函数原型:_CRTIMP int __cdecl feof(FILE *);

    fputc()函数:将字符逐一写入文件中

    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
  7. 4.3字符串存取函数fputs()/fgets()

    函数原型: _CRTIMP int __cdecl fputs(const char*,FILE *);

               _CRTIMP char* __cdecl fgets(char *,int, FILE *);

    fgets()函数:从指定文件读入一个字符串,如fgets(str,n,fp);

    函数参数:n为要求得到的字符串个数,但只从fp指向的文件输入n-1个字符,然后最后加一个‘\0’字符,因此共得到n个字符的字符串,把他们放在字符数组str中。如果在读完n-1个字符之前,遇到换行符或EOF,读取结束。

    fgets()函数:向指定文件输出一个字符串,如fputs(“Hey”,fp);把字符串Hey输出到fp指定文件。

    函数参数:第一个参数可以是字符串常量、字符数组或字符型指针。

    返回值:输出成功,返回0;否则返回EOF;

    C/C++:文本文件和二进制文件的读写
  8. 5、格式化存取函数

    函数原型: _CRTIMP int __cdecl fprintf(FILE*,const char *,…);

              ....... _CRTIMP int __cdecl fscanf(FILE*,const char *,…);

    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写
  9. 6、二进制文件操作

    ①指针重返函数

    函数原型:_CRTIMP void __cdecl rewind(FILE *);

    函数功能:使位置指针重返文件的开头,用于文件的定位。

    ②fread() /fwrite()

    函数原型: _CRTIMP size_t __cdecl fread(void*,size_t,size_t,FILE *);

               _CRTIMP size_t __cdecl fwrite(const void*,size_t,size_t,FILE*);

    调用形式:fread(buffer,size,count,fp);

              fwrite(buffer,size,count,fp);

    参数:buffer:读入或输出数据的地址;

          size:读写输入时,每组数据的大小;

          cout:读写数据的次数;

          fp:文件指针;

    函数功能:一次读取一组数据,可以读取count次;

    示例:

    #include <stdio.h>

    #define SIZE 3

    typedef enum {MM,GG} Gender;

    typedef struct

    {

    char name[10];

    int age;

    Gender gender;

    }Person;

    void write2file(Person emp[SIZE])

    {

    FILE *fp;

    if((fp=fopen(“emp.txt”,”wb”))==NULL)

    {

        printf(“cannot open file! \n”);

        return;

    }

    for(int i=0;i<SIZE;i++)

        if(fwrite(&emp[i],sizeof(Person),1,fp) != 1)

            printf(“file write error! \n”);

    fclose(fp);

    }

    void read_from_file(FILE *fp)

    {

    Person emp_out[SIZE];

    if((fp=fopen(“emp.txt”,”rb”))==NULL)

    {

        printf(“cannot open file! \n”);

        return;

    }

    printf(“\n%d employee’s information read: \n”,SIZE);

    for(int i=0;i<SIZE;i++)

    {

        if(fread(&emp_out[i],sizeof(Person),1,fp)!=1)

            if(feof(fp))

            {

                 fclose(fp);

    return;

    }

    printf(“%-5s %4d %5d \n”,emp_out[i].name, emp_out[i].age,emp_out[i].gender);

    }

    fclose(fp);

    }

     

    int main()

    {

    FILE *fp=NULL;

    Person employee[SIZE];

    printf(“Enter %d employee’s information: \n”,SIZE);

    for(int i=0;i<SIZE;i++)

        scanf(“%s %d %d”, employee[i].name, &employee[i].age, &employee[i].gender);

    write2file(employee);

    read_from_file(fp);

        return 0;

    }

  10. 7、随机存取函数fseek()

    函数原型: _CRTIMP int __cdecl fseek(FILE*,long,int);

    流式文件可以顺序读写,也可以随机读写。

    关键在于控制文件的位置指针,

    如果位置指针是按字节位置顺序移动的,就是顺序读写;

    如果位置指针按需要移到到任意位置,就可实现随机读写。

    所谓随机读写,是指读完上一个字符字节后,并不一定要读写其后续的字符字节,而可以读写文件中任意位置上需要的字符字节。

    函数调用形式:fseek(fp,offset,start);

    参数:

    start:起始点,用0,1,2代替,0表示文件开始,名字为SEEK_SET,1表示当前位置,名字为SEEK_CUR,2表示文件末尾,名字为SEEK_END。

    fseek()函数一般用于二进制文件,因为文本文件要发生字符转换,计算位置会发生混乱。

    示例;

    fseek(fp,i*sizeof(Person),0);

    C/C++:文本文件和二进制文件的读写
    C/C++:文本文件和二进制文件的读写

猜你喜欢

转载自blog.csdn.net/u010312937/article/details/79401550