上一节中鸡啄米讲了定时器Timer的用法,本节介绍下文件操作类CFile类的使用。

       CFile类概述

       如果你学过C语言,应该知道文件操作使用的是文件指针,通过文件指针实现对它指向的文件的各种操作。这些文件操作函数中有的最终还是调用了操作系统的API函数或者处理过程与之类似,例如在Windows系统中,fread函数就调用了API函数ReadFile。

       Windows系统的API函数除了ReadFile,还有CreateFile、WriteFile等函数。而MFC基于面向对象的思想,将这些Windows API函数封装到了CFile类中,实现对文件的打开、关闭、读、写、获取文件信息等操作。使用CFile类对文件进行操作非常便捷。

       CFile类的成员函数

       CFile( );
       CFile(HANDLE hFile);
       CFile(LPCTSTR lpszFileName,UINT nOpenFlags);

       以上三个成员函数都是CFile的构造函数,用于构造CFile对象。参数hFile为要关联到CFile对象的文件的句柄。参数lpszFileName为要关联到CFile对象的文件的相对路径或者绝对路径;参数nOpenFlags为文件访问选项的组合,通过各选项的按位或运算实现组合,下面的5个表列出了nOpenFlags参数可能取的选项:

       下面的文件访问模式选项表中只能选择一个进行组合,默认取CFile::modeRead。

取值描述
CFile::modeRead只读方式访问文件
CFile::modeWrite写入方式访问文件
CFile::modeReadWrite读写方式访问文件

       下面的文件共享模式选项表中也只能选择一个进行组合,默认的共享模式是CFile::shareExclusive。

取值描述
CFile::shareDenyNone允许其他进程对文件进行读写
CFile::shareDenyRead不允许其他进程读取文件
CFile::shareDenyWrite不允许其他进程写文件
CFile::shareExclusive禁止其他进程对文件的所有访问

       下面的文件创建模式选项列表中可选择第一个或两者都选进行组合。

取值描述
CFile::modeCreate如果文件不存在则创建文件,而如果存在则将它关联到此CFile对象并将长度截取为0
CFile::modeNoTruncate如果文件不存在则创建文件,而如果存在则将它关联到此CFile对象而不进行截取

       注意,选择CFile::modeNoTruncate时需要与CFile::modeCreate一起使用,即CFile::modeCreate | CFile::modeNoTruncate。

       另外,还有一个文件缓冲选项列表和一个文件安全选项。文件缓冲选项不太常用,鸡啄米这里就不讲了,有兴趣的可以查阅MSDN。文件安全选项是CFile::modeNoInherit,意为禁止子进程继承使用此文件。

       当然,在实际使用时,以上各个表并不是都要用到,大家可以根据自己的需要选择用哪个表,选择哪个选项。

       virtual BOOL Open(LPCTSTR lpszFileName,UINT nOpenFlags,CFileException* pError = NULL);

       打开文件。它通常与默认构造函数CFile::CFile()一起使用。参数lpszFileName和nOpenFlags同构造函数。参数pError为指向文件异常对象的指针,默认为NULL。

       virtual void Close( );

       关闭文件。如果你没有在执行析构函数前调用此成员函数关闭文件,则析构函数会为你关闭。

       virtual UINT Read(void* lpBuf,UINT nCount);

       读取文件数据到缓存。参数lpBuf是由用户提供的指向接收文件数据的缓存的指针;参数nCount为读取的最大字节数。返回值是实际读取到缓存的字节数,如果到达文件尾则返回值可能会小于nCount,此时继续读取的话,会返回0,所以通常我们都会判断返回值是否小于nCount或者等于0来确定是否到达文件尾。

       virtual void Write(const void* lpBuf,UINT nCount);

       将缓存中的数据写入文件。参数lpBuf也是由用户提供,指向包含写入数据的缓存的指针;参数nCount为缓存中要被写入文件的数据的字节数。

       virtual ULONGLONG Seek(LONGLONG lOff,UINT nFrom);

       在一个打开的文件中重定位文件指针。参数lOff为文件指针移动的字节个数,为正数时表示向文件尾移动,为负数时表示向文件开头移动;参数nFrom为lOff的基准位置,即由nFrom位置开始移动lOff个字节,它可以取下面几个值中的一个:

       CFile::begin       从文件开头开始移动
       CFile::current    从文件指针的当前位置开始移动 
       CFile::end          从文件尾开始移动

       文件打开时,文件指针被置于0,即文件开头处。

       如果此函数成功则返回文件指针的位置。

       void SeekToBegin( );

       将文件指针移动到文件开头。它等价于Seek( 0L, CFile::begin )。

       ULONGLONG SeekToEnd( );

       将文件指针移动到文件末尾。返回值是文件的字节长度。它等价于CFile::Seek( 0L, CFile::end )。

       virtual ULONGLONG GetLength( ) const;

       获取文件的字节长度。

       virtual void SetLength(ULONGLONG dwNewLen);

       改变文件的长度。参数dwNewLen为文件的新长度,它可能比文件的当前长度值要大或者小,文件会相应的被扩展或截取。

       virtual CString GetFileName( ) const;

       获取文件名称。

       virtual CString GetFilePath( ) const;

       获取文件的绝对路径。

       virtual CString GetFileTitle( ) const;

       获取文件的显示名称。举个例子,与GetFileName区分一下,如果你系统中的文件不显示扩展名,则它获取到的文件名称就不包含扩展名,否则就显示扩展名。

       virtual ULONGLONG GetPosition( ) const;

       获取文件指针的当前位置。

       static void PASCAL Remove(LPCTSTR lpszFileName,CAtlTransactionManager* pTM = NULL);

       删除文件。参数lpszFileName为要删除的文件路径,可以是相对路径、绝对路径或者网络路径;参数pTM指向一个CAtlTransactionManager对象。

       static void PASCAL Rename(LPCTSTR lpszOldName,LPCTSTR lpszNewName,CAtlTransactionManager* pTM = NULL);

       重命名文件。参数lpszOldName为老的文件路径;参数lpszNewName为新的文件路径;参数pTM指向一个CAtlTransactionManager对象。实际上此函数的意义已经不只是重命名文件,还可以移动文件到其他目录下,例如,lpszOldName取"d:\\1.txt",lpszNewName取"e:\\2.txt",这样可以将D盘中的1.txt文件转移到E盘并重命名为2.txt。

       CFile类应用实例

       这里鸡啄米只给大家演示几个简单的代码片段,从这些代码片段中熟悉CFile类的文件操作。

       实例一:构造CFile对象时就打开文件,然后向文件中写入数据,最后以Seek函数移动文件指针,读取文件内容。

char writeBuffer[500];     // 要写入的数据的缓存   
char readBuffer[500];      // 存放读取数据的缓存   
LONGLONG lOff = 0;         // 文件指针的偏移量,也是读取到的数据的总字节数  
// 构造CFile对象,同时以创建和读写的方式打开文件E:\1.txt   
CFile file(_T("e:\\1.txt"), CFile::modeCreate | CFile::modeReadWrite);   
  
// 将写入数据的缓存中每个字节都赋值为字符c   
memset(writeBuffer, 'c', sizeof(writeBuffer));   
// 将数据写入到文件中   
file.Write(writeBuffer, sizeof(writeBuffer));   
  
while (true)   
{   
    // 以文件开头为基准,移动文件指针到lOff的位置   
    file.Seek(lOff, CFile::begin);   
    // 读取100个字节的数据到存放读取数据的缓存的readBuffer + lOff位置处   
    int nRet = file.Read(readBuffer + lOff, 100);   
    // 根据实际读取的字节数,增加文件指针的移动量   
    lOff += nRet;   
    // 如果读取数据时返回值小于指定的100,说明已到文件尾,跳出循环   
    if (nRet < 100)   
        break;   
}   
  
// 关闭文件   
file.Close();

     实际上,在Write函数和Read函数执行后,文件指针会自动移动到最后操作的位置,所以其实上面的代码中无须使用Seek函数再去手动移动文件指针。这将在下面的实例二中体现出来。

       实例二:构造CFile对象,然后使用Open成员函数打开文件,再写入一个结构体数组,最后读取出来。

       先贴上结构体的定义:

struct student   
{   
    int  nNum;     
    float fScore;   
};

     下面是文件操作的代码片段:

student s1[2];   // 存放要写入文件的数据   
student s2[2];   // 存放从文件读取的数据   
CFile file;      // CFile对象   
int nReadBytes = 0;   // 从文件中读取到的总字节数   
  
// 为s1数组各元素赋值   
s1[0].nNum = 22;   
s1[0].fScore = 91.5;   
s1[1].nNum = 23;   
s1[1].fScore = 85;   
  
// 以创建、读写方式打开文件E:\1.txt   
if (file.Open(_T("E:\\1.txt"), CFile::modeCreate | CFile::modeReadWrite))   
{   
    // 写入数据s1结构体数组   
    file.Write(s1, sizeof(s1));   
    // 因为上面调用Write以后文件指针在文件尾,所以需要将其移动到文件开头   
    file.SeekToBegin();   
  
    while (true)   
    {   
        // 读取数据到s2   
        int nRet = file.Read((BYTE*)s2 + nReadBytes, sizeof(student));   
        // 计算已经读取到的总字节数   
        nReadBytes += nRet;   
        // 如果读取数据时返回值小于指定的sizeof(student),则说明已到文件尾,跳出循环   
        if (nRet < sizeof(student))   
            break;   
    }   
  
    // 关闭文件   
    file.Close();   
}

       本节内容就到这里,如果有其他语言的文件操作的经验的话,应该还是比较简单的。鸡啄米很高兴能在大家的编程入门之路上贡献自己一点微薄的力量。