• VC的文件操作


    各种关于文件的操作在程序设计中是十分常见,如果能对其各种操作都了如指掌,就可以根据实际情况找到最佳的解决方案,从而在较短的时间内编写出高效的代码,因而熟练的掌握文件操作是十分重要的。本文将对Visual C++中有关文件操作进行全面的介绍,并对在文件操作中经常遇到的一些疑难问题进行详细的分析。


      1.文件的查找 
      当对一个文件操作时,如果不知道该文件是否存在,就要首先进行查找。MFC中有一个专门用来进行文件查找的类CFileFind,使用它可以方便快捷地进行文件的查找。下面这段代码演示了这个类的最基本使用方法。 
      CString strFileTitle; 
      CFileFind finder; 
      BOOL bWorking = finder.FindFile("C:\windows\sysbkup\*.cab"); 
      while(bWorking) 
      { 
      bWorking=finder.FindNextFile(); 
      strFileTitle=finder.GetFileTitle(); 
      }


      2.文件的打开/保存对话框 
      让用户选择文件进行打开和存储操作时,就要用到文件打开/保存对话框。MFC的类CFileDialog用于实现这种功能。使用CFileDialog声明一个对象时,第一个BOOL型参数用于指定文件的打开或保存,当为TRUE时将构造一个文件打开对话框,为FALSE时构造一个文件保存对话框。 
      在构造CFileDialog对象时,如果在参数中指定了OFN_ALLOWMULTISELECT风格,则在此对话框中可以进行多选操作。此时要重点注意为此CFileDialog对象的m_ofn.lpstrFile分配一块内存,用于存储多选操作所返回的所有文件路径名,如果不进行分配或分配的内存过小就会导致操作失败。下面这段程序演示了文件打开对话框的使用方法。 
      CFileDialog mFileDlg(TRUE,NULL,NULL, 
      OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT|OFN_ALLOWMULTISELECT, 
      "All Files (*.*)|*.*||",AfxGetMainWnd()); 
      CString str(" ",10000); 
      mFileDlg.m_ofn.lpstrFile=str.GetBuffer(10000); 
      str.ReleaseBuffer(); 
      POSITION mPos=mFileDlg.GetStartPosition(); 
      CString pathName(" ",128); 
      CFileStatus status; 
      while(mPos!=NULL) 
      { 
      pathName=mFileDlg.GetNextPathName(mPos); 
      CFile::GetStatus( pathName, status ); 
      }


      3.文件的读写 
      文件的读写非常重要,下面将重点进行介绍。文件读写的最普通的方法是直接使用CFile进行,如文件的读写可以使用下面的方法: 
      //对文件进行读操作 
      char sRead[2]; 
      CFile mFile(_T("user.txt"),CFile::modeRead); 
      if(mFile.GetLength()<2) 
      return; 
      mFile.Read(sRead,2); 
      mFile.Close(); 
      //对文件进行写操作 
      CFile mFile(_T("user.txt "), CFile::modeWrite|CFile::modeCreate); 
      mFile.Write(sRead,2); 
      mFile.Flush(); 
      mFile.Close(); 
      虽然这种方法最为基本,但是它的使用繁琐,而且功能非常简单。我向你推荐的是使用CArchive,它的使用方法简单且功能十分强大。首先还是用CFile声明一个对象,然后用这个对象的指针做参数声明一个CArchive对象,你就可以非常方便地存储各种复杂的数据类型了。它的使用方法见下例。 
      //对文件进行写操作 
      CString strTemp; 
      CFile mFile; 
      mFile.Open("d:\dd\try.TRY",CFile::modeCreate|CFile::modeNoTruncate|CFile::modeWrite); 
      CArchive ar(&mFile,CArchive::store); 
      ar<<  ar.Close(); 
      mFile.Close(); 
      //对文件进行读操作 
      CFile mFile; 
      if(mFile.Open("d:\dd\try.TRY",CFile::modeRead)==0) 
      return; 
      CArchive ar(&mFile,CArchive::load); 
       ar>>strTemp; 
      ar.Close(); 
      mFile.Close(); 
      CArchive的 << 和>> 操作符用于简单数据类型的读写,对于CObject派生类的对象的存取要使用ReadObject()和WriteObject()。使用CArchive的ReadClass()和WriteClass()还可以进行类的读写,如: 
      //存储CAboutDlg类 
      ar.WriteClass(RUNTIME_CLASS(CAboutDlg)); 
      //读取CAboutDlg类 
      CRuntimeClass* mRunClass=ar.ReadClass(); 
      //使用CAboutDlg类 
      CObject* pObject=mRunClass->CreateObject(); 
      ((CDialog* )pObject)->DoModal(); 
      虽然VC提供的文档/视结构中的文档也可进行这些操作,但是不容易理解、使用和管理,因此虽然很多VC入门的书上花费大量篇幅讲述文档/视结构,但我建议你最好不要使用它的文档。关于如何进行文档/视的分离有很多书介绍,包括非常著名的《Visual C++ 技术内幕》。 
      如果你要进行的文件操作只是简单的读写整行的字符串,我建议你使用CStdioFile,用它来进行此类操作非常方便,如下例。 
      CStdioFile mFile; 
      CFileException mExcept; 
      mFile.Open( "d:\temp\aa.bat", CFile::modeWrite, &mExcept); 
      CString string="I am a string."; 
      mFile.WriteString(string); 
      mFile.Close();


     4.临时文件的使用 

      正规软件经常用到临时文件,你经常可以会看到C:WindowsTemp目录下有大量的扩展名为tmp的文件,这些就是程序运行是建立的临时文件。临时文件的使用方法基本与常规文件一样,只是文件名应该调用函数GetTempFileName()获得。它的第一个参数是建立此临时文件的路径,第二个参数是建立临时文件名的前缀,第四个参数用于得到建立的临时文件名。得到此临时文件名以后,你就可以用它来建立并操作文件了,如: 
      char szTempPath[_MAX_PATH],szTempfile[_MAX_PATH]; 
      GetTempPath(_MAX_PATH, szTempPath); 
      GetTempFileName(szTempPath,_T ("my_"),0,szTempfile); 
      CFile m_tempFile(szTempfile,CFile:: modeCreate|CFile:: modeWrite); 
      char m_char='a'; 
      m_tempFile.Write(&m_char,2); 
      m_tempFile.Close(); 
      5.文件的复制、删除等 
      MFC中没有提供直接进行这些操作的功能,因而要使用SDK。SDK中的文件相关函数常用的有CopyFile()、CreateDirectory()、DeleteFile()、MoveFile()。它们的用法很简单,可参考MSDN。


    *********************************************************************************************************************
    ×××××××××××××××××××××××××××××××××××××××××××××××××
    *********************************************************************************************************************
    如何进行文件操作

    [1]显示对话框,取得文件名

    CString FilePathName;
    CFileDialog dlg(TRUE);///TRUE为OPEN对话框,FALSE为SAVE AS对话框
    if (dlg.DoModal() == IDOK)
    FilePathName=dlg.GetPathName();

    相关信息:CFileDialog 用于取文件名的几个成员函数:
    假如选择的文件是C:WINDOWSTEST.EXE
    则(1)GetPathName();取文件名全称,包括完整路径。取回C:WINDOWSTEST.EXE
    (2)GetFileTitle();取文件全名:TEST.EXE
    (3)GetFileName();取回TEST
    (4)GetFileExt();取扩展名EXE

    [2]打开文件
    CFile file("C:HELLO.TXT",CFile::modeRead);//只读方式打开
    //CFile::modeRead可改为 CFile::modeWrite(只写),
    //CFile::modeReadWrite(读写),CFile::modeCreate(新建)
    例子:
    {
    CFile file;
    file.Open("C:HELLO.TXT",CFile::modeCreate|Cfile::modeWrite);
    .
    .
    .
    }

    [3]移动文件指针
    file.Seek(100,CFile::begin);///从文件头开始往下移动100字节
    file.Seek(-50,CFile::end);///从文件末尾往上移动50字节
    file.Seek(-30,CFile::current);///从当前位置往上移动30字节
    file.SeekToBegin();///移到文件头
    file.SeekToEnd();///移到文件尾

    [4]读写文件
    读文件:
    char buffer[1000];
    file.Read(buffer,1000);
    写文件:
    CString string("自强不息");
    file.Write(string,8);

    [5]关闭文件
    file.Close();

     

     

    在我们写的程序当中,总有一些配置信息需要保存下来,以便完成程序的功能,最简单的办法就是将这些信息写入INI文件中,程序初始化时再读入.具体应用如下:

      一.将信息写入.INI文件中.

      1.所用的WINAPI函数原型为:

    BOOL WritePrivateProfileString( 
    LPCTSTR lpAppName, 
    LPCTSTR lpKeyName, 
    LPCTSTR lpString, 
    LPCTSTR lpFileName 
    );

      其中各参数的意义:

       LPCTSTR lpAppName 是INI文件中的一个字段名.

       LPCTSTR lpKeyName 是lpAppName下的一个键名,通俗讲就是变量名.

       LPCTSTR lpString 是键值,也就是变量的值,不过必须为LPCTSTR型或CString型的.

       LPCTSTR lpFileName 是完整的INI文件名.

      2.具体使用方法:设现有一名学生,需把他的姓名和年龄写入 c:studstudent.ini 文件中.

    CString strName,strTemp; 
    int nAge; 
    strName="张三"; 
    nAge=12; 
    ::WritePrivateProfileString("StudentInfo","Name",strName,"c:\stud\student.ini");

      此时c:studstudent.ini文件中的内容如下:

       [StudentInfo] 
       Name=张三

      3.要将学生的年龄保存下来,只需将整型的值变为字符型即可:

    strTemp.Format("%d",nAge); 
    ::WritePrivateProfileString("StudentInfo","Age",strTemp,"c:\stud\student.ini");


     二.将信息从INI文件中读入程序中的变量.

      1.所用的WINAPI函数原型为:

    DWORD GetPrivateProfileString( 
    LPCTSTR lpAppName, 
    LPCTSTR lpKeyName, 
    LPCTSTR lpDefault, 
    LPTSTR lpReturnedString, 
    DWORD nSize, 
    LPCTSTR lpFileName 
    );

      其中各参数的意义:

       前二个参数与 WritePrivateProfileString中的意义一样.

       lpDefault : 如果INI文件中没有前两个参数指定的字段名或键名,则将此值赋给变量.

       lpReturnedString : 接收INI文件中的值的CString对象,即目的缓存器.

       nSize : 目的缓存器的大小.

       lpFileName : 是完整的INI文件名.

      2.具体使用方法:现要将上一步中写入的学生的信息读入程序中.

    CString strStudName; 
    int nStudAge; 
    GetPrivateProfileString("StudentInfo","Name","默认姓名",strStudName.GetBuffer(MAX_PATH),MAX_PATH,"c:\stud\student.ini");

      执行后 strStudName 的值为:"张三",若前两个参数有误,其值为:"默认姓名".

      3.读入整型值要用另一个WINAPI函数:

    UINT GetPrivateProfileInt( 
    LPCTSTR lpAppName, 
    LPCTSTR lpKeyName, 
    INT nDefault, 
    LPCTSTR lpFileName 
    );

      这里的参数意义与上相同.使用方法如下:

    nStudAge=GetPrivateProfileInt("StudentInfo","Age",10,"c:\stud\student.ini");


    三.循环写入多个值,设现有一程序,要将最近使用的几个文件名保存下来,具体程序如下:

      1.写入:

    CString strTemp,strTempA; 
    int i; 
    int nCount=6; 
    file://共有6个文件名需要保存 
    for(i=0;i {strTemp.Format("%d",i); 
    strTempA=文件名; 
    file://文件名可以从数组,列表框等处取得. 
    ::WritePrivateProfileString("UseFileName","FileName"+strTemp,strTempA, 
    "c:\usefile\usefile.ini"); 
    } 
    strTemp.Format("%d",nCount); 
    ::WritePrivateProfileString("FileCount","Count",strTemp,"c:\usefile\usefile.ini"); 
    file://将文件总数写入,以便读出.

      2.读出:

    nCount=::GetPrivateProfileInt("FileCount","Count",0,"c:\usefile\usefile.ini"); 
    for(i=0;i {strTemp.Format("%d",i); 
    strTemp="FileName"+strTemp; 
    ::GetPrivateProfileString("CurrentIni",strTemp,"default.fil", strTempA.GetBuffer(MAX_PATH),MAX_PATH,"c:\usefile\usefile.ini");

    file://使用strTempA中的内容.

    }

      补充四点: 
       1.INI文件的路径必须完整,文件名前面的各级目录必须存在,否则写入不成功,该函数返回 FALSE 值. 
       2.文件名的路径中必须为 \ ,因为在VC++中, \ 才表示一个 . 
       3.也可将INI文件放在程序所在目录,此时 lpFileName 参数为: ".\student.ini".

     //---------------------------------------------------------------------------------- 
    /* 
    类名:CIni 
    版本:v2.0 
    最后更新: 
    v2.0 
    梦小孩于2004年2月14日情人节 
    加入高级操作的功能 
    v1.0 
    梦小孩于2003年某日 
    一般操作完成 

    类描述: 
    本类可以于.ini文件进行操作 
    */ 

    文件 1: 

    #pragma once 

    #include "afxTempl.h" 

    class CIni 
    { 
    private: 
    CString m_strFileName; 
    public: 
    CIni(CString strFileName):m_strFileName(strFileName) 
    { 
    } 
    public: 
    //一般性操作: 
    BOOL SetFileName(LPCTSTR lpFileName); //设置文件名 
    CString GetFileName(void); //获得文件名 
    BOOL SetValue(LPCTSTR lpSection, LPCTSTR lpKey, LPCTSTR lpValue,bool bCreate=true); //设置键值,bCreate是指段名及键名未存在时,是否创建。 
    CString GetValue(LPCTSTR lpSection, LPCTSTR lpKey); //得到键值. 
    BOOL DelSection(LPCTSTR strSection); //删除段名 
    BOOL DelKey(LPCTSTR lpSection, LPCTSTR lpKey); //删除键名 


    public: 
    //高级操作: 
    int GetSections(CStringArray& arrSection); //枚举出全部的段名 
    int GetKeyValues(CStringArray& arrKey,CStringArray& arrValue,LPCTSTR lpSection); //枚举出一段内的全部键名及值 

    BOOL DelAllSections(); 

    }; 

    文件 2: 

    #include "StdAfx.h" 
    #include "ini.h" 

    #define MAX_ALLSECTIONS 2048 //全部的段名 
    #define MAX_SECTION 260 //一个段名长度 
    #define MAX_ALLKEYS 6000 //全部的键名 
    #define MAX_KEY 260 //一个键名长度 

    BOOL CIni::SetFileName(LPCTSTR lpFileName) 
    { 
    CFile file; 
    CFileStatus status; 

    if(!file.GetStatus(lpFileName,status)) 
    return TRUE; 

    m_strFileName=lpFileName; 
    return FALSE; 
    } 

    CString CIni::GetFileName(void) 
    { 
    return m_strFileName; 
    } 

    BOOL CIni::SetValue(LPCTSTR lpSection, LPCTSTR lpKey, LPCTSTR lpValue,bool bCreate) 
    { 
    TCHAR lpTemp[MAX_PATH] ={0}; 

    //以下if语句表示如果设置bCreate为false时,当没有这个键名时则返回TRUE(表示出错) 
    //!*&*none-value*&!* 这是个垃圾字符没有特别意义,这样乱写是防止凑巧相同。 
    if (!bCreate) 
    { 
    GetPrivateProfileString(lpSection,lpKey,"!*&*none-value*&!*",lpTemp,MAX_PATH,m_strFileName); 
    if(strcmp(lpTemp,"!*&*none-value*&!*")==0) 
    return TRUE; 
    } 

    if(WritePrivateProfileString(lpSection,lpKey,lpValue,m_strFileName)) 
    return FALSE; 
    else 
    return GetLastError(); 
    } 

    CString CIni::GetValue(LPCTSTR lpSection, LPCTSTR lpKey) 
    { 
    DWORD dValue; 
    TCHAR lpValue[MAX_PATH] ={0}; 

    dValue=GetPrivateProfileString(lpSection,lpKey,"",lpValue,MAX_PATH,m_strFileName); 
    return lpValue; 
    } 

    BOOL CIni::DelSection(LPCTSTR lpSection) 
    { 
    if(WritePrivateProfileString(lpSection,NULL,NULL,m_strFileName)) 
    return FALSE; 
    else 
    return GetLastError(); 
    } 

    BOOL CIni::DelKey(LPCTSTR lpSection, LPCTSTR lpKey) 
    { 
    if(WritePrivateProfileString(lpSection,lpKey,NULL,m_strFileName)) 
    return FALSE; 
    else 
    return GetLastError(); 
    } 


    int CIni::GetSections(CStringArray& arrSection) 
    { 
    /* 
    本函数基础: 
    GetPrivateProfileSectionNames - 从 ini 文件中获得 Section 的名称 
    如果 ini 中有两个 Section: [sec1] 和 [sec2],则返回的是 'sec1',0,'sec2',0,0 ,当你不知道 
    ini 中有哪些 section 的时候可以用这个 api 来获取名称 
    */ 
    int i; 
    int iPos=0; 
    int iMaxCount; 
    TCHAR chSectionNames[MAX_ALLSECTIONS]={0}; //总的提出来的字符串 
    TCHAR chSection[MAX_SECTION]={0}; //存放一个段名。 
    GetPrivateProfileSectionNames(chSectionNames,MAX_ALLSECTIONS,m_strFileName); 

    //以下循环,截断到两个连续的0 
    for(i=0;i<MAX_ALLSECTIONS;i++) 
    { 
    if (chSectionNames[i]==0) 
    if (chSectionNames[i]==chSectionNames[i+1]) 
    break; 
    } 

    iMaxCount=i+1; //要多一个0号元素。即找出全部字符串的结束部分。 
    arrSection.RemoveAll();//清空原数组 

    for(i=0;i<iMaxCount;i++) 
    { 
    chSection[iPos++]=chSectionNames[i]; 
    if(chSectionNames[i]==0) 
    { 
    arrSection.Add(chSection); 
    memset(chSection,0,MAX_SECTION); 
    iPos=0; 
    } 

    } 

    return (int)arrSection.GetSize(); 
    } 

    int CIni::GetKeyValues(CStringArray& arrKey,CStringArray& arrValue, LPCTSTR lpSection) 
    { 
    /* 
    本函数基础: 
    GetPrivateProfileSection- 从 ini 文件中获得一个Section的全部键名及值名 
    如果ini中有一个段,其下有 "段1=值1" "段2=值2",则返回的是 '段1=值1',0,'段2=值2',0,0 ,当你不知道 
    获得一个段中的所有键及值可以用这个。 
    */ 
    int i; 
    int iPos=0; 
    CString strKeyValue; 
    int iMaxCount; 
    TCHAR chKeyNames[MAX_ALLKEYS]={0}; //总的提出来的字符串 
    TCHAR chKey[MAX_KEY]={0}; //提出来的一个键名 

    GetPrivateProfileSection(lpSection,chKeyNames,MAX_ALLKEYS,m_strFileName); 

    for(i=0;i<MAX_ALLKEYS;i++) 
    { 
    if (chKeyNames[i]==0) 
    if (chKeyNames[i]==chKeyNames[i+1]) 
    break; 
    } 

    iMaxCount=i+1; //要多一个0号元素。即找出全部字符串的结束部分。 
    arrKey.RemoveAll();//清空原数组 
    arrValue.RemoveAll(); 

    for(i=0;i<iMaxCount;i++) 
    { 
    chKey[iPos++]=chKeyNames[i]; 
    if(chKeyNames[i]==0) 
    { 
    strKeyValue=chKey; 
    arrKey.Add(strKeyValue.Left(strKeyValue.Find("="))); 
    arrValue.Add(strKeyValue.Mid(strKeyValue.Find("=")+1)); 
    memset(chKey,0,MAX_KEY); 
    iPos=0; 
    } 

    } 

    return (int)arrKey.GetSize(); 
    } 

    BOOL CIni::DelAllSections() 
    { 
    int nSection; 
    CStringArray arrSection; 
    nSection=GetSections(arrSection); 
    for(int i=0;i<nSection;i++) 
    { 
    if(DelSection(arrSection[i])) 
    return GetLastError(); 
    } 
    return FALSE; 
    } 


    使用方法: 
    CIni ini("c:\a.ini"); 
    int n; 

    /*获得值 
    TRACE("%s",ini.GetValue("段1","键1")); 
    */ 

    /*添加值 
    ini.SetValue("自定义段","键1","值"); 
    ini.SetValue("自定义段2","键1","值",false); 
    */ 

    /*枚举全部段名 
    CStringArray arrSection; 
    n=ini.GetSections(arrSection); 
    for(int i=0;i<n;i++) 
    TRACE("%s ",arrSection[i]); 
    */ 

    /*枚举全部键名及值 
    CStringArray arrKey,arrValue; 
    n=ini.GetKeyValues(arrKey,arrValue,"段1"); 
    for(int i=0;i<n;i++) 
    TRACE("键:%s 值:%s ",arrKey[i],arrValue[i]); 
    */ 

    /*删除键值 
    ini.DelKey("段1","键1"); 
    */ 

    /*删除段 
    ini.DelSection("段1"); 
    */ 

    /*删除全部 
    ini.DelAllSections(); 
    */

     

    VC++中以追加方式向文本文件写入数据

     

    在VB、Asp中向文本文件追加数据很容易,只要设定一个参数为ForAppending就行了。

    Sub OpenTextFileTest

       Const ForReading = 1, ForWriting = 2, ForAppending = 8

       Dim fso, f

       Set fso = CreateObject("Scripting.FileSystemObject")

       Set f = fso.OpenTextFile("c: estfile.txt", ForWriting, True)

       f.Write "Hello world!"

       f.Close

    End Sub


    在c语言中,追加数据也比较简单,好像设定a+参数就可以了。

     

    今天,我要用MFC中的CStdioFile类进行文件操作,读写等。

    可是,看了下好像没有简单的方法,

    于是在网上看到这样的写法:

    CStdioFile file(strFile,CFile::modeCreate|CFile::modeNoTruncate|CFile::modeWrite);

    file.WriteString(strTmp);

    file.Close;


    modeNoTruncate的意思就是不要截取的意思吧

    可是,试了下这段代码,并没有起作用,不知道是什么原因。

    于是,在WriteString写字符串之前加了个把指针先定位到文件末尾的代码,就可以了

    CString strTmp="hehe ";
     
    CStdioFile file(strFile,CFile::modeCreate|CFile::modeNoTruncate|CFile::modeWrite);

    file.SeekToEnd();//先定位到文件尾部

    file.WriteString(strTmp);

    file.Close;

     

    随着Windows 2000和XP的普及,现在的大文件越来越多,而VC6中MFC的CFile类只支持不大于4GB的文件, 原因在于CFile类中使用了32位整型来处理文件,32位数的范围是2的32次方(4GB),超过这个范围的文件CFile就管不了,微软.Net中VC7的CFile类支持大于4GB的文件,而.Net还不普及,开发桌面应用VC6还是首选,所以我们可以参照VC7写一个CFile的继承类CFile64,使它支持大于4GB的文件: 
    class CFile64 : public CFile
    {
    public:

    // Attributes
     ULONGLONG GetPosition();


    // Overridables

     virtual ULONGLONG Seek(LONGLONG lOff, UINT nFrom);
     virtual void SetLength(ULONGLONG dwNewLen);
     ULONGLONG GetLength() ;

     virtual void LockRange(ULONGLONG dwPos, ULONGLONG dwCount);
     virtual void UnlockRange(ULONGLONG dwPos, ULONGLONG dwCount);


    };


    #include "stdafx.h"
    #include "file64.h"

    ////////////////////////////////////////////////////////////////////////////
    // CFile64 implementation


    ULONGLONG CFile64::Seek(LONGLONG lOff, UINT nFrom)
    {
     ASSERT_VALID(this);
     ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
     ASSERT(nFrom == begin || nFrom == end || nFrom == current);
     ASSERT(begin == FILE_BEGIN && end == FILE_END && current == FILE_CURRENT);

       LARGE_INTEGER liOff;

       liOff.QuadPart = lOff;
     liOff.LowPart = ::SetFilePointer((HANDLE)m_hFile, liOff.LowPart, &liOff.HighPart,
       (DWORD)nFrom);
     if (liOff.LowPart  == (DWORD)-1)
       if (::GetLastError() != NO_ERROR)
         CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);

     return liOff.QuadPart;
    }

    ULONGLONG CFile64::GetPosition() 
    {
     ASSERT_VALID(this);
     ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);

       LARGE_INTEGER liPos;
       liPos.QuadPart = 0;
     liPos.LowPart = ::SetFilePointer((HANDLE)m_hFile, liPos.LowPart, &liPos.HighPart , FILE_CURRENT);
     if (liPos.LowPart == (DWORD)-1)
       if (::GetLastError() != NO_ERROR)
         CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);

     return liPos.QuadPart;
    }

    void CFile64::LockRange(ULONGLONG dwPos, ULONGLONG dwCount)
    {
     ASSERT_VALID(this);
     ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);

       ULARGE_INTEGER liPos;
       ULARGE_INTEGER liCount;

       liPos.QuadPart = dwPos;
       liCount.QuadPart = dwCount;
     if (!::LockFile((HANDLE)m_hFile, liPos.LowPart, liPos.HighPart, liCount.LowPart, 
       liCount.HighPart))
       {
      CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
       }
    }

    void CFile64::UnlockRange(ULONGLONG dwPos, ULONGLONG dwCount)
    {
     ASSERT_VALID(this);
     ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);

       ULARGE_INTEGER liPos;
       ULARGE_INTEGER liCount;

       liPos.QuadPart = dwPos;
       liCount.QuadPart = dwCount;
     if (!::UnlockFile((HANDLE)m_hFile, liPos.LowPart, liPos.HighPart, liCount.LowPart,
       liCount.HighPart))
       {
      CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
       }
    }

    void CFile64::SetLength(ULONGLONG dwNewLen)
    {
     ASSERT_VALID(this);
     ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);

     Seek(dwNewLen, (UINT)begin);

     if (!::SetEndOfFile((HANDLE)m_hFile))
      CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
    }

    ULONGLONG CFile64::GetLength() 
    {
     ASSERT_VALID(this);

       ULARGE_INTEGER liSize;
       liSize.LowPart = ::GetFileSize((HANDLE)m_hFile, &liSize.HighPart);
       if (liSize.LowPart == (DWORD)-1)
       if (::GetLastError() != NO_ERROR)
       CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);

     return liSize.QuadPart;
    }


    /////////////////////////////////////////////////////////////////////////////

    LONGLONG是64位整型,这样在理论上可支持的最大文件为18000000000GB,你也可以根据自己的需要重载CFile的其他函数

  • 相关阅读:
    3.学习SpringMVC注解深入
    1.Google Blockly概述以及入门
    4.Servlet的Response和Request
    3.Servlet的ServletContext对象
    提高网络点击率的八种方法
    JNDI知道这么多就够了!
    jetty源代码剖析
    cocos2d-x 源代码分析 : control 源代码分析 ( 控制类组件 controlButton)
    Codeforces Round #254 (Div. 2)B. DZY Loves Chemistry
    几款“理想”“习惯”应用体验报告
  • 原文地址:https://www.cnblogs.com/lidabo/p/3470085.html
Copyright © 2020-2023  润新知