07.C语言文件操作

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

1. 使用文件的原因

我们前面学习结构体时写了通讯录的程序当通讯录运行起来的时候可以给通讯录中增加、删除数据此时数据是存放在内存中当程序退出的时候通讯录中的数据自然就不存在了等下次运行通讯录程序的时候数据又得重新录入如果使用这样的通讯录就很难受。
我们在想既然是通讯录就应该把信息记录下来只有我们自己选择删除数据的时候数据才不复存在。这就涉及到了数据持久化的问题我们一般数据持久化的方法有把数据存放在磁盘文件、存放到数据库等方式。
使用文件我们可以将数据直接存放在电脑的硬盘上做到了 数据的持久化

2. 文件

磁盘上的文件是文件。

但是在程序设计中我们一般谈的文件有两种程序文件、数据文件从文件功能的角度来分类的。

2.1 程序文件

包括源程序文件后缀为.c,目标文件windows环境后缀为.obj,可执行程序windows环境后缀为.exe。

2.2 数据文件

文件的内容不一定是程序而是程序运行时读写的数据比如程序运行需要从中读取数据的文件或者输出内容的文件。

本章讨论的是数据文件。

在以前各章所处理数据的输入输出都是以终端为对象的即从终端的键盘输入数据运行结果显示到显示器上。

其实有时候我们会把信息输出到磁盘上当需要的时候再从磁盘上把数据读取到内存中使用这里处理的就是磁盘上文件。

2.3 文件名

一个文件要有一个唯一的文件标识以便用户识别和引用。

文件名包含3部分文件路径+文件名主干+文件后缀

例如 c:\code\test.txt

为了方便起见文件标识常被称为文件名

3. 文件的打开和关闭

3.1 文件指针

缓冲文件系统中关键的概念是“文件类型指针”简称“文件指针”。

每个被使用的文件都在内存中开辟了一个相应的文件信息区用来存放文件的相关信息如文件的名

字文件状态及文件当前的位置等。这些信息是保存在一个结构体变量中的。该结构体类型是有系统

声明的取名FILE.

例如VS2013编译环境提供的 stdio.h 头文件中有以下的文件类型申明

struct _iobuf {
    char* _ptr;
    int _cnt;
    char* _base;
    int _flag;
    int _file;
    int _charbuf;
    int _bufsiz;
    char* _tmpfname;
};
typedef struct _iobuf FILE;
//只要打开文件就会创建文件信息区

不同的C编译器的FILE类型包含的内容不完全相同但是大同小异。

每当打开一个文件的时候系统会根据文件的情况自动创建一个FILE结构的变量并填充其中的信息

使用者不必关心细节。

一般都是通过一个FILE的指针来维护这个FILE结构的变量这样使用起来更加方便。

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

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

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

比如

3.2 文件的打开和关闭

文件在读写之前应该先打开文件在使用结束之后应该关闭文件

在编写程序的时候在打开文件的同时都会返回一个FILE*的指针变量指向该文件也相当于建立了指

针和文件的关系。

ANSIC 规定使用fopen函数来打开文件fclose来关闭文件。

//打开文件
FILE * fopen ( const char * filename, const char * mode );//读的名字打开方式
//关闭文件
int fclose ( FILE * stream );

打开方式如下

文件使用方式

含义

如果指定文件不存在

“r”只读

为了输入数据打开一个已经存在的文本文件

出错

“w”只写

为了输出数据打开一个文本文件//会将上一个文件销毁掉

建立一个新的文件

“a”追加

向文本文件尾添加数据

建立一个新的文件

“rb”只读

为了输入数据打开一个二进制文件

出错

“wb”只写

为了输出数据打开一个二进制文件

建立一个新的文件

“ab”追加

向一个二进制文件尾添加数据

出错

“r+”读写

为了读和写打开一个文本文件

出错

“w+”读写

为了读和写建议一个新的文件

建立一个新的文件

“a+”读写

打开一个文件在文件尾进行读写

建立一个新的文件

“rb+”读写

为了读和写打开一个二进制文件

出错

“wb+”读写

为了读和写新建一个新的二进制文件

建立一个新的文件

“ab+”读写

打开一个二进制文件在文件尾进行读和写

建立一个新的文件

实例代码

int main()
{
    //D:\\code\\test.txt - 绝对路径
    //
    FILE* pf = fopen("test2.txt", "w");//此时文件路径的c语言代码路径一直不用加前缀此时为相对路径
    if (pf == NULL)
    {
        perror("fopen");
        return 1;
    }
    else
    {
        printf("打开文件成功\n");
    }
    //读文件
    //....

    //关闭文件
    fclose(pf);
    pf = NULL;

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

4. 文件的顺序读写

功能

函数名

适用于

字符输入函数

fgetc

所有输入流

字符输出函数

fputc

所有输出流

文本行输入函数

fgets

所有输入流

文本行输出函数

fputs

所有输出流

格式化输入函数

fscanf

所有输入流

格式化输出函数

fprintf

所有输出流

二进制输入

fread

文件

二进制输出

fwrite

文件

4.1 对比一组函数

scanf/fscanf/sscanf
printf/fprintf/sprintf
scanf一从键盘上读取格式化的数据stdin
printf -把数据写到(输出)屏幕.上stdout
fscanf -针对所有输入流的格式化的输入函数: stdin, 打开的文件
fprintf -针对所有输出流的格式化的输出函数: stdout, 打开的文件
sscanf - 从一个字符串中还原出一个格式化的数据
sprintf -把格式化的数据存放在(转换成) - 1个字符串结构体转化成字符串
struct S
{
    char name[20];
    int age;
    float score;
};

int main()
{
    struct S s = {"zhangsan", 20, 98.5};
    char buf[100] = { 0 };
    sprintf(buf, "%s %d %f", s.name, s.age, s.score);
    printf("%s\n", buf);//按照字符串打印的

    struct S tmp = { 0 };
    sscanf(buf, "%s %d %f", tmp.name, &(tmp.age), &(tmp.score));
    printf("%s %d %f\n", tmp.name, tmp.age, tmp.score);//打印结构体数据

    return 0;
}

4.2 fputc*fgetc函数

int main()
{
    FILE* pf = fopen("test.txt", "w");
    if (NULL == pf)
    {
        perror("fopen");
        return 1;
    }
    //写文件
    /*fputc('a', pf);
    fputc('b', pf);
    fputc('c', pf);
    fputc('d', pf);*/

    char ch = 0;
    for (ch = 'a'; ch <= 'z'; ch++)
    {
        fputc(ch, pf);
    }

    //关闭文件
    fclose(pf);
    pf = NULL;

    return 0;
}

4.2 fgets&fputs函数

说要读5个实际上字符为4个因为存在\0

int main()
{
    FILE* pf = fopen("test.txt", "r");
    if (NULL == pf)
    {
        perror("fopen");
        return 1;
    }
    //读文件
    int ch = fgetc(pf);
    printf("%c\n", ch);

    ch = fgetc(pf);
    printf("%c\n", ch);

    ch = fgetc(pf);
    printf("%c\n", ch);

    //关闭文件
    fclose(pf);
    pf = NULL;

    return 0;
}
int main()
{
    FILE* pf = fopen("test.txt", "w");
    if (NULL == pf)
    {
        perror("fopen");
        return 1;
    }
    //写文件
    //测试写一行数据
    fputs("hello world\n", pf);
    fputs("hello bit\n", pf);

    return 0;
}
int main()
{
    FILE* pf = fopen("test.txt", "r");
    if (NULL == pf)
    {
        perror("fopen");
        return 1;
    }
    //读文件
    //测试一行数据
    char buf[20] = {0};
    fgets(buf, 20, pf);
    printf("%s", buf);
    fgets(buf, 20, pf);
    printf("%s", buf);
     //关闭文件
    fclose(pf);
    pf = NULL;
    return 0;
}

4.3 fscanf&fprintf函数

printf是可变参数列表和fprintf雷同

struct S
{
    char name[20];
    int age;
    float score;
};

int main()
{
    struct S s = { "zhangsan", 20, 95.5 };
    FILE* pf = fopen("test.txt", "w");
    if (NULL == pf)
    {
        perror("fopen");
        return 1;
    }
    //格式化的写入文件
    fprintf(pf, "%s %d %f\n", s.name, s.age, s.score);//s.name不用取地址符是因为数组本身就是地址
    //关闭文件
    fclose(pf);
    pf = NULL;
    return 0;
}

//任何一个C语言程序运行的时候: 默认打开3个流标准输入(键盘)

类型:

stdin -标准输入键盘 FILE*

stdout - 标准输出(屏幕) FILE*

stderr标准错误(屏幕) FILE*

//上述函数既可以针对文件又可以针对标准输入输出流

struct S
{
    char name[20];
    int age;
    float score;
};

int main()
{
    struct S s = { 0 };
    fscanf(stdin, "%s %d %f", s.name, &(s.age), &(s.score));
    fprintf(stdout, "%s %d %f\n", s.name, s.age, s.score);

    //int ch = fgetc(stdin);
    //fputc(ch, stdout);

    return 0;
}

5. 文件的随机读写

5.1 fseek

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

int fseek ( FILE * stream, long int offset, int origin );//offset偏移量

例子

/* fseek example */
#include <stdio.h>
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;
}

5.2 ftell

返回文件指针相对于起始位置的偏移量。
long int ftell ( FILE * stream );

EG:

/* ftell example : getting size of a file */
#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;
}

5.3 rewind

让文件指针的位置回到文件的起始位置
void rewind ( FILE * stream );

EG:

/* rewind example */
#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';
    puts(buffer);
    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;
}

7. 文件读取结束的判定

7.1 被错误使用的feof

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

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

1. 文本文件读取是否结束判断返回值是否为 EOF fgetc 或者 NULL fgets

例如

fgetc 判断是否为 EOF .读取成功返回得到的字符读取失败返回EOF

fgets 判断返回值是否为 NULL .

2. 二进制文件的读取结束判断判断返回值是否小于实际要读的个数。

例如

fread判断返回值是否小于实际要读的个数。

正确的使用

文本文件的例子

#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);
}
首先文件读取结束了(想知道是遇到错误结束还是读取失败结束)
结束后想知道读取结束的原因:
feof - 返回真就说明是文件正常读取遇到了结束标志而结束的。
ferror -返回真就说明是文件在读取过程中出错了而结束

二进制文件的例子

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

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);//单位毫秒其实是10s
    fclose(pf);
    //注fclose在关闭文件的时候也会刷新缓冲区
    pf = NULL;
    return 0;
}

这里可以得出一个结论

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

件。

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

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6