c语言_文件操作_FILE结构体解释_涉及对操作系统文件FCB操作的解释_

简介:

1. 文件和流的关系

 

  C将每个文件简单地作为顺序字节流(如下图)。每个文件用文件结束符结束,或者在特定字节数的地方结束,这个特定的字节数可以存储在系统维护的管理数据结构中。当打开文件时,就建立了和文件的关系。

  在开始执行程序的时候,将自动打开3个文件和相关的流:标准输入流、标准输出流和标准错误。流提供了文件和程序的通信通道。例如,标准输入流使得程序可以从键盘读取数据,而标准输出流使得程序可以在屏幕上输出数据。打开一个文件将返回指向FILE结构(在stdio.h中定义)的指针,它包含用于处理文件的信息,也就是说,这个结构包含文件描述符。文件描述符是操作系统数组(打开文件列表的索引)。每个数组元素包含一个文件控制块(FCB, File Control Block),操作系统用它来管理特定的文件。

  标准输入、标准输出和标准错误是用文件指针stdin、stdout和stderr来处理的。

 

2. C语言文件操作的底层实现简介

 

2.1 FILE结构体

 

C语言的stdio.h头文件中,定义了用于文件操作的结构体FILE。这样,我们通过fopen返回一个文件指针(指向FILE结构体的指针)来进行文件操作。可以在stdio.h(位于visual studio安装目录下的include文件夹下)头文件中查看FILE结构体的定义,如下:

 

复制代码
 
TC2.0中:
typedef struct  {
        short           level;          /* fill/empty level of buffer */
        unsigned        flags;          /* File status flags    */
        char            fd;             /* File descriptor      */
        unsigned char   hold;           /* Ungetc char if no buffer */
        short           bsize;          /* Buffer size          */
        unsigned char   *buffer;        /* Data transfer buffer */
        unsigned char   *curp;          /* Current active pointer */
        unsigned        istemp;         /* Temporary file indicator */
        short           token;          /* Used for validity checking */
}       FILE;                           /* This is the FILE object */




VC6.0中:
#ifndef _FILE_DEFINED
struct _iobuf {

    char *_ptr; //文件输入的下一个位置
int _cnt; //当前缓冲区的相对位置
char *_base; //指基础位置(即是文件的其始位置) 
int _flag; //文件标志
int _file; //文件的有效性验证
int _charbuf; //检查缓冲区状况,如果无缓冲区则不读取
int _bufsiz; //???这个什么意思
char *_tmpfname; //临时文件名

        };
typedef struct _iobuf FILE;
#define _FILE_DEFINED
#endif
复制代码

 

2.2 C语言文件管理的实现

 

C程序用不同的FILE结构管理每个文件。程序员可以使用文件,但是不需要知道FILE结构的细节。实际上,FILE结构是间接地操作系统的文件控制块
(FCB)来实现对文件的操作的,如下图:

 

上面图中的_file实际上是一个描述符,作为进入打开文件表索引的整数。

 

2.3 操作系统文件管理简介

 

从2.2中的图可以看出,C语言通过FILE结构可以间接操作文件控制块(FCB)。为了加深对这些的理解,这里科普下操作系统对打开文件的管理。

文件是存放在物理磁盘上的,包括文件控制块(FCB)和数据块。文件控制块通常包括文件权限、日期(创建、读取、修改)、拥有者、文件大小、数据块信息。数据块用来存储实际的内容。对于打开的文件,操作系统是这样管理的:

1
系统维护了两张表,一张是系统级打开文件表,一张是进程级打开文件表(每个进程有一个)。

系统级打开文件表复制了文件控制块的信息等;进程级打开文件表保存了指向系统级文件表的指针及其他信息。

系统级文件表每一项都保存一个计数器,即该文件打开的次数。我们初次打开一个文件时,系统首先查看该文件是否已在系统级文件表中,如果不在,则创建该项信息,否则,计数器加1。当我们关闭一个文件时,相应的计数也会减1,当减到0时,系统将系统级文件表中的项删除。

进程打开一个文件时,会在进程级文件表中添加一项。每项的信息包括当前文件偏移量(读写文件的位置)、存取权限、和一个指向系统级文件表中对应文件项的指针。系统级文件表中的每一项通过文件描述符(一个非负整数)来标识。

联系2.2和2.3上面的内容,可以发现,应该是这样的:FILE结构体中的_file成员应该是指向进程级打开文件表,然后,通过进程级打开文件表可以找到系统级打开文件表,进而可以通过FCB操作物理磁盘上面的文件。

 

2.4 文件操作的例子

 

复制代码
#include <stdio.h>  
void main() {  
    FILE * fp1;  
    FILE * fp2;  
    fp1 = fopen("filetest.cpp","r");  
    fp2 = fopen("filetest.cpp","r");  
    

  char buffer[256];
fscanf(fp1,
"%s",buffer); printf("%s\n",buffer);
fscanf(fp2,
"%s",buffer); printf("%s\n",buffer);
printf(
"fp1(filetest.cpp):%d\n",fp1->_file); printf("fp2(filetest.cpp):%d\n",fp2->_file); printf("stdin:%d\n",stdin->_file); printf("stdout:%d\n",stdout->_file); printf("stderr:%d\n",stderr->_file); }
复制代码

 

filetest.cpp中的内容如下:

 

复制代码
#include<stdio.h>  
int main()  
{  
    printf("Hello World!\n");  
    return 0;  
}  
复制代码

 

运行结果如下:

 


通过这个程序可以看出,应该是每打开一次文件,哪怕多次打开的都是同一个文件,进程级打开文件表中应该都会添加一个记录。如果是打开的是同一个文件,这多条记录对应着同一个物理磁盘文件。由于每一次打开文件所进行的操作都是通过进程级打开文件表中不同的记录来实现的,这样,相当于每次打开文件的操作是相对独立的,这就是上面的程序的运行结果中,两次读取文件的结果是一样的(而不是第二次读取从第一次结束的位置进行)。

另外,还可以看出,程序运行的时候,默认三个流是打开的stdin,stdout和stderr,它们的_file描述符分别是0、1和2。也可以看出,该程序打开的文件描述符依次从3开始递增。

 

3.顺序访问文件

 

3.1 顺序写入文件

 

先看一个例子:

复制代码
#include <stdio.h>  
int main()  
{  
    int account;//账号  
    char name[30];//账号名  
    double balance;//余额  
  
    FILE *cfPtr;  
    if ((cfPtr=fopen("clients.dat","w"))==NULL)  
    {  
        printf("File could not be opened.\n");  
    }  
    else  
    {  
        printf("Enter the account, name and the balance:\n");  
        printf("Enter EOF to end input.\n");  
        printf("? ");  
        scanf("%d%s%lf",&account,name,&balance);  
        while(!feof(stdin))  
        {  
            fprintf(cfPtr,"%d %s %.2f\n",account,name,balance);  
            printf("? ");  
            scanf("%d%s%lf",&account,name,&balance);  
        }  
        fclose(cfPtr);  
    }  
    return 0;  
}  
复制代码

 

 

运行结果:

 

从上面的例子中可以看出,写入文件大致需两步:定义文件指针和打开文件

函数fopen有两个参数:文件名和文件打开模式。文件打开模式‘w’说明文件时用于写入的。如果以写入模式打开的文件不存在,则fopen将创建该文件。如果打开现有的文件来写入,则将抛弃文件原有的内容而没有任何警告。在程序中,if语句用于确定文件指针cfPtr是否是NULL(没有成功打开文件时fopen的返回值)。如果是NULL,则将输出错误消息,然后程序终止。否则,处理输入并写入到文件中。

foef(stdin)用来确定用户是否从标准输入输入了文件结束符。文件结束符通知程序没有其他数据可以处理了。foef的参数是指向测试是否为文件结束符的FILE指针。一旦输入了文件结束符,函数将返回一个非零值;否则,函数返回0。当没有输入文件结束符时,程序继续执行while循环。

fprintf(cfPtr,"%d %s %.2f\n",account,name,balance);向文件clients.dat中写入数据。稍后通过用于读取文件的程序,就可以提取数据。函数fprintf和printf等价,只是fprintf还需要一个指向文件的指针,所有数据都写入到这个文件中。

在用户输入文件结束之后,程序用fclose关闭clients.dat文件,并结束运行。函数fclose也接收文件指针作为参数。如果没有明确地调用函数fclose,则操作系统通常在程序执行结束的稍后关闭文件。这是操作系统“内务管理”的一个示例,但是,这样可能会带来一些难以预料的问题,所以一定要注意在使用结束之后关闭文件。

 

3.2 文件打开模式

 

模式 说明
r 打开文件,进行读取。
w 创建文件,以进行写入。如果文件已经存在,则删除当前内容。
a 追加,打开或创建文件以在文件尾部写入。
r+ 打开文件以进行更新(读取和写入)。
w+ 创建文件以进行更新。如果文件已经存在,则删除当前内容。
a+ 追加,打开或者创建文件以进行更新,在文件尾部写入。

3.3 顺序读取文件

 

下面的例子读取的是上一个例子中写入数据生成的文件。

复制代码
#include <stdio.h>  
int main()  
{  
    int account;//账号  
    char name[30];//账号名  
    double balance;//余额  
  
    FILE *cfPtr;  
    if ((cfPtr=fopen("clients.dat","r"))==NULL)  
    {  
        printf("File could not be opened.\n");  
    }  
    else  
    {  
        printf("%-10s%-13s%s\n","Account","Name","Balance");  
        fscanf(cfPtr,"%d%s%lf",&account,name,&balance);  
        while(!feof(cfPtr))  
        {  
            printf("%-10d%-13s%lf\n",account,name,balance);  
            fscanf(cfPtr,"%d%s%lf",&account,&name,&balance);  
        }  
        fclose(cfPtr);  
    }  
    return 0;  
}  
复制代码

运行结果:

 



上面的例子中,只需将第一个例子中的文件打开模式从w变为r,就可以打开文件读取数据。

同样地,fscanf(cfPtr,"%d%s%lf",&account,name,&balance);函数从文件中读取一条记录。函数fscanf和函数scanf等价看,只是fscanf接收将从中读取数据的文件指针作为参数。在第一次执行前面的语句时,account的值为100,name的值是Jones,而balance等于24.98。每次执行第二条fscanf语句时,将从文件中读取另一条记录,而account,name和balance将有新值。当到达文件结束位置时,关闭文件,而程序终止。

要从文件中顺序检索数据,程序通常从文件的开始来读取,而且连续读取所有数据,直至找到期望的数据。在程序执行过程中,有可能会多次处理文件中的数据(重新从文件的开头处理数据)。这时候就要用到函数rewind(cfPtr);,它可以使程序的文件位置指针(表示文件中将要读取或者写入的下一个字节的位置)重新设置到文件的开头(也就是偏移量为0的字节)。注意,文件位置指针并不是指针,它是指定文件中将进行下一次读取或者写入的位置的整数值,有时候也称其为文件偏移量,它是FILE结构的成员。

 

4.随机访问文件

 

文件中用格式化输入函数fprintf所创建的记录的长度并不是完全一致的。然而,在随机访问文件中,单个记录的长度通常是固定的,而且可以直接访问(这样速度更快)而无需通过其他记录来查找。这使得随机文件访问适合飞机订票系统,银行系统,销售点系统和其他需要快速访问特定数据的事务处理系统。我们可以有很多方法来实现随机访问文件,但是这里我们将把讨论的范围限制在使用固定长度记录的简单方法上。

函数fwrite把从内存中特定位置开始的指定数量的字节写入到文件位置指针指定的文件位置,函数fread从文件位置指针指定的文件位置处把指定数量的字节复制到指定的内存位置。fwrite和fread可以从磁盘上读取数据数组,以及向磁盘上写入数据数组。fread和fwrite的第三个参数是从磁盘中读取或者写入到磁盘上的数组元素的个数。

 

文件处理程序很少向文件中写入字段。通常情况下,它们一次写入一个struct。

 

4.1 创建随机访问的文件

 

复制代码
#include<stdio.h>  
struct clientData  
{  
    int acctNum;  
    char lastName[15];  
    char firstName[10];  
    double balance;  
};  
int main()  
{  
    int i;  
    struct clientData blankClient={0,"","",0.0};  
    FILE *cfPtr;  
    if ((cfPtr = fopen("credit.dat","wb"))== NULL)  
    {  
        printf("File could not be opened.\n");  
    }  
    else  
    {  
        for (i=1;i<=100;i++)  
        {  
            fwrite(&blankClient,sizeof(struct clientData),1,cfPtr);  
        }  
        fclose(cfPtr);  
    }  
    return 0;  
}  
复制代码

 

fwrite(&blankClient,sizeof(struct clientData),1,cfPtr);用于向文件中写入一个数据块,其会在cfPtr指向的文件中写入大小为sizeof(struct clientData)的结构blankClient。当然,也可以写入对象数组的多个元素,只需把数组名传给第一个参数,把要写入的元素个数写入第三个参数即可。

 

4.2 随机向随机访问文件中写入数据

 

复制代码
#include<stdio.h>  
struct clientData  
{  
    int acctNum;  
    char lastName[15];  
    char firstName[10];  
    double balance;  
};  
int main()  
{  
    int i;  
    struct clientData client={0,"","",0.0};  
    FILE *cfPtr;  
    if ((cfPtr = fopen("credit.dat","rb+"))== NULL)  
    {  
        printf("File could not be opened.\n");  
    }  
    else  
    {  
        printf("Enter account number(1 to 100, 0 to end input\):\n");  
        scanf("%d",&client.acctNum);  
        while (client.acctNum!=0)  
        {  
            printf("Enter lastname, firstname, balance\n");  
            fscanf(stdin,"%s%s%lf",client.lastName,client.firstName,&client.balance);  
            //在文件中定位用户指定的记录  
            fseek(cfPtr,(client.acctNum-1)*sizeof(struct clientData),SEEK_SET);  
            //将用户指定的信息写入文件  
            fwrite(&client,sizeof(struct clientData),1,cfPtr);  
  
            //输入下一个账号  
            printf("Enter account number:\n");  
            scanf("%d",&client.acctNum);  
        }  
        fclose(cfPtr);  
    }  
    return 0;  
}  
复制代码

 

 

运行结果:

 



fseek(cfPtr,(client.acctNum-1)*sizeof(struct clientData),SEEK_SET);将cfPtr所引用文件的位置指针移动到由(client.acctNum-1)*sizeof(struct clientData)计算所得到的字节位置处,这个表达式的值称为偏移量或者位移。负号常量SEEK_SET说明,文件位置指针指向的位置是相对于文件开头的偏移量。

ANSI标准制定了fseek的函数原型为int fseek(FILE *stream, long int offset, int whence);其中offset是stream指向的文件中从位置whence开始的字节数。参数whence可以有三个值:SEEK_SET, SEEKCUR或者SEEK_END,分别对应文件的开头当前位置和结尾。


4.2 从随机访问文件中读取数据

 

复制代码
#include<stdio.h>  
struct clientData  
{  
    int acctNum;  
    char lastName[15];  
    char firstName[10];  
    double balance;  
};  
int main()  
{  
    struct clientData client={0,"","",0.0};  
    FILE *cfPtr;  
    if ((cfPtr = fopen("credit.dat","rb"))== NULL)  
    {  
        printf("File could not be opened.\n");  
    }  
    else  
    {  
        printf("%-6s%-16s%-11s%10s\n","Acct","Last name","First name","Balance");  
        while(!feof(cfPtr))  
        {  
            fread(&client,sizeof(struct clientData),1,cfPtr);  
            if (client.acctNum!=0)  
            {  
                printf("%-6d%-16s%-11s%10.2f\n",client.acctNum,client.lastName,client.firstName,client.balance);  
            }  
        }  
        fclose(cfPtr);  
    }  
    return 0;  
}  
复制代码

 

运行结果:

 

本文转自二郎三郎博客园博客,原文链接:http://www.cnblogs.com/haore147/p/3648395.html,如需转载请自行联系原作者

相关文章
|
12天前
|
存储 小程序 C语言
【C语言程序设计——文件】文件操作(头歌实践教学平台习题)【合集】
本文介绍了C语言中的文件操作,分为两个关卡。第1关任务是将键盘输入的字符(以#结束)存入`file1.txt`并显示输出;第2关任务是从键盘输入若干行文本(每行不超过80个字符,用-1作为结束标志),写入`file2.txt`后再读取并显示。文中详细讲解了文件的打开、读取(使用`fgetc()`和`fgets()`)、写入(使用`fputc()`和`fputs()`)及关闭操作,并提供了示例代码和测试说明。
31 5
|
1月前
|
存储 网络协议 编译器
【C语言】深入解析C语言结构体:定义、声明与高级应用实践
通过根据需求合理选择结构体定义和声明的放置位置,并灵活结合动态内存分配、内存优化和数据结构设计,可以显著提高代码的可维护性和运行效率。在实际开发中,建议遵循以下原则: - **模块化设计**:尽可能封装实现细节,减少模块间的耦合。 - **内存管理**:明确动态分配与释放的责任,防止资源泄漏。 - **优化顺序**:合理排列结构体成员以减少内存占用。
167 14
|
1月前
|
存储 编译器 C语言
【C语言】结构体详解 -《探索C语言的 “小宇宙” 》
结构体通过`struct`关键字定义。定义结构体时,需要指定结构体的名称以及结构体内部的成员变量。
199 10
|
2月前
|
存储 数据建模 程序员
C 语言结构体 —— 数据封装的利器
C语言结构体是一种用户自定义的数据类型,用于将不同类型的数据组合在一起,形成一个整体。它支持数据封装,便于管理和传递复杂数据,是程序设计中的重要工具。
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
218 13
|
2月前
|
存储 编译器 数据处理
C 语言结构体与位域:高效数据组织与内存优化
C语言中的结构体与位域是实现高效数据组织和内存优化的重要工具。结构体允许将不同类型的数据组合成一个整体,而位域则进一步允许对结构体成员的位进行精细控制,以节省内存空间。两者结合使用,可在嵌入式系统等资源受限环境中发挥巨大作用。
88 11
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
179 3
|
2月前
|
Python
文件元数据获取方法对比:`os.path` 与 `os.stat`
本文对比了Python中两种获取文件元数据的方法:`os.path`和`os.stat`。通过示例代码展示了如何获取文件大小和修改时间,并从性能、功能性和代码可读性三方面进行了详细对比。最终给出了根据具体需求选择合适方法的最佳实践建议。
37 2
|
2月前
|
存储 人工智能 算法
数据结构实验之C 语言的函数数组指针结构体知识
本实验旨在复习C语言中的函数、数组、指针、结构体与共用体等核心概念,并通过具体编程任务加深理解。任务包括输出100以内所有素数、逆序排列一维数组、查找二维数组中的鞍点、利用指针输出二维数组元素,以及使用结构体和共用体处理教师与学生信息。每个任务不仅强化了基本语法的应用,还涉及到了算法逻辑的设计与优化。实验结果显示,学生能够有效掌握并运用这些知识完成指定任务。
70 4
|
3月前
|
存储 C语言
如何在 C 语言中实现结构体的深拷贝
在C语言中实现结构体的深拷贝,需要手动分配内存并逐个复制成员变量,确保新结构体与原结构体完全独立,避免浅拷贝导致的数据共享问题。具体方法包括使用 `malloc` 分配内存和 `memcpy` 或手动赋值。
105 10

热门文章

最新文章