VC++中文件操作(二)--- .ini文件、CFile64

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

 

VC++中文件操作(二)

 ***************************************************************************
××××××××××第一、VC有关文件操作,参考资料2
***************************************************************************

在我们写的程序当中,总有一些配置信息需要保存下来,以便完成程序的功能,最简单的办法就是将这些信息写入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:\stud\student.ini 文件中.

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

  此时c:\stud\student.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".

 

 1  //---------------------------------------------------------------------------------- 
 2 /* 
 3 类名:CIni 
 4 版本:v2.0 
 5 最后更新: 
 6 v2.0 
 7 梦小孩于2004年2月14日情人节 
 8 加入高级操作的功能 
 9 v1.0 
10 梦小孩于2003年某日 
11 一般操作完成 
12 
13 类描述: 
14 本类可以于.ini文件进行操作 
15 */ 
16 
17 文件 1: 
18 
19 #pragma once 
20 
21 #include "afxTempl.h" 
22 
23 class CIni 
24 { 
25 private: 
26 CString m_strFileName; 
27 public: 
28 CIni(CString strFileName):m_strFileName(strFileName) 
29 { 
30 } 
31 public: 
32 //一般性操作: 
33 BOOL SetFileName(LPCTSTR lpFileName); //设置文件名 
34 CString GetFileName(void); //获得文件名 
35 BOOL SetValue(LPCTSTR lpSection, LPCTSTR lpKey, LPCTSTR lpValue,bool bCreate=true); //设置键值,bCreate是指段名及键名未存在时,是否创建。 
36 CString GetValue(LPCTSTR lpSection, LPCTSTR lpKey); //得到键值. 
37 BOOL DelSection(LPCTSTR strSection); //删除段名 
38 BOOL DelKey(LPCTSTR lpSection, LPCTSTR lpKey); //删除键名 
39 
40 
41 public: 
42 //高级操作: 
43 int GetSections(CStringArray& arrSection); //枚举出全部的段名 
44 int GetKeyValues(CStringArray& arrKey,CStringArray& arrValue,LPCTSTR lpSection); //枚举出一段内的全部键名及值 
45 
46 BOOL DelAllSections(); 
47 
48 }; 

 

 

 文件 2: 

  1 #include "StdAfx.h" 
  2 #include "ini.h" 
  3 
  4 #define MAX_ALLSECTIONS 2048 //全部的段名 
  5 #define MAX_SECTION 260 //一个段名长度 
  6 #define MAX_ALLKEYS 6000 //全部的键名 
  7 #define MAX_KEY 260 //一个键名长度 
  8 
  9 BOOL CIni::SetFileName(LPCTSTR lpFileName) 
 10 { 
 11 CFile file; 
 12 CFileStatus status; 
 13 
 14 if(!file.GetStatus(lpFileName,status)) 
 15 return TRUE; 
 16 
 17 m_strFileName=lpFileName; 
 18 return FALSE; 
 19 } 
 20 
 21 CString CIni::GetFileName(void) 
 22 { 
 23 return m_strFileName; 
 24 } 
 25 
 26 BOOL CIni::SetValue(LPCTSTR lpSection, LPCTSTR lpKey, LPCTSTR lpValue,bool bCreate) 
 27 { 
 28 TCHAR lpTemp[MAX_PATH] ={0}; 
 29 
 30 //以下if语句表示如果设置bCreate为false时,当没有这个键名时则返回TRUE(表示出错) 
 31 //!*&*none-value*&!* 这是个垃圾字符没有特别意义,这样乱写是防止凑巧相同。 
 32 if (!bCreate) 
 33 { 
 34 GetPrivateProfileString(lpSection,lpKey,"!*&*none-value*&!*",lpTemp,MAX_PATH,m_strFileName); 
 35 if(strcmp(lpTemp,"!*&*none-value*&!*")==0) 
 36 return TRUE; 
 37 } 
 38 
 39 if(WritePrivateProfileString(lpSection,lpKey,lpValue,m_strFileName)) 
 40 return FALSE; 
 41 else 
 42 return GetLastError(); 
 43 } 
 44 
 45 CString CIni::GetValue(LPCTSTR lpSection, LPCTSTR lpKey) 
 46 { 
 47 DWORD dValue; 
 48 TCHAR lpValue[MAX_PATH] ={0}; 
 49 
 50 dValue=GetPrivateProfileString(lpSection,lpKey,"",lpValue,MAX_PATH,m_strFileName); 
 51 return lpValue; 
 52 } 
 53 
 54 BOOL CIni::DelSection(LPCTSTR lpSection) 
 55 { 
 56 if(WritePrivateProfileString(lpSection,NULL,NULL,m_strFileName)) 
 57 return FALSE; 
 58 else 
 59 return GetLastError(); 
 60 } 
 61 
 62 BOOL CIni::DelKey(LPCTSTR lpSection, LPCTSTR lpKey) 
 63 { 
 64 if(WritePrivateProfileString(lpSection,lpKey,NULL,m_strFileName)) 
 65 return FALSE; 
 66 else 
 67 return GetLastError(); 
 68 } 
 69 
 70 
 71 int CIni::GetSections(CStringArray& arrSection) 
 72 { 
 73 /* 
 74 本函数基础: 
 75 GetPrivateProfileSectionNames - 从 ini 文件中获得 Section 的名称 
 76 如果 ini 中有两个 Section: [sec1] 和 [sec2],则返回的是 'sec1',0,'sec2',0,0 ,当你不知道 
 77 ini 中有哪些 section 的时候可以用这个 api 来获取名称 
 78 */ 
 79 int i; 
 80 int iPos=0; 
 81 int iMaxCount; 
 82 TCHAR chSectionNames[MAX_ALLSECTIONS]={0}; //总的提出来的字符串 
 83 TCHAR chSection[MAX_SECTION]={0}; //存放一个段名。 
 84 GetPrivateProfileSectionNames(chSectionNames,MAX_ALLSECTIONS,m_strFileName); 
 85 
 86 //以下循环,截断到两个连续的0 
 87 for(i=0;i<MAX_ALLSECTIONS;i++) 
 88 { 
 89 if (chSectionNames[i]==0) 
 90 if (chSectionNames[i]==chSectionNames[i+1]) 
 91 break; 
 92 } 
 93 
 94 iMaxCount=i+1; //要多一个0号元素。即找出全部字符串的结束部分。 
 95 arrSection.RemoveAll();//清空原数组 
 96 
 97 for(i=0;i<iMaxCount;i++) 
 98 { 
 99 chSection[iPos++]=chSectionNames[i]; 
100 if(chSectionNames[i]==0) 
101 { 
102 arrSection.Add(chSection); 
103 memset(chSection,0,MAX_SECTION); 
104 iPos=0; 
105 } 
106 
107 } 
108 
109 return (int)arrSection.GetSize(); 
110 } 
111 
112 int CIni::GetKeyValues(CStringArray& arrKey,CStringArray& arrValue, LPCTSTR lpSection) 
113 { 
114 /* 
115 本函数基础: 
116 GetPrivateProfileSection- 从 ini 文件中获得一个Section的全部键名及值名 
117 如果ini中有一个段,其下有 "段1=值1" "段2=值2",则返回的是 '段1=值1',0,'段2=值2',0,0 ,当你不知道 
118 获得一个段中的所有键及值可以用这个。 
119 */ 
120 int i; 
121 int iPos=0; 
122 CString strKeyValue; 
123 int iMaxCount; 
124 TCHAR chKeyNames[MAX_ALLKEYS]={0}; //总的提出来的字符串 
125 TCHAR chKey[MAX_KEY]={0}; //提出来的一个键名 
126 
127 GetPrivateProfileSection(lpSection,chKeyNames,MAX_ALLKEYS,m_strFileName); 
128 
129 for(i=0;i<MAX_ALLKEYS;i++) 
130 { 
131 if (chKeyNames[i]==0) 
132 if (chKeyNames[i]==chKeyNames[i+1]) 
133 break; 
134 } 
135 
136 iMaxCount=i+1; //要多一个0号元素。即找出全部字符串的结束部分。 
137 arrKey.RemoveAll();//清空原数组 
138 arrValue.RemoveAll(); 
139 
140 for(i=0;i<iMaxCount;i++) 
141 { 
142 chKey[iPos++]=chKeyNames[i]; 
143 if(chKeyNames[i]==0) 
144 { 
145 strKeyValue=chKey; 
146 arrKey.Add(strKeyValue.Left(strKeyValue.Find("="))); 
147 arrValue.Add(strKeyValue.Mid(strKeyValue.Find("=")+1)); 
148 memset(chKey,0,MAX_KEY); 
149 iPos=0; 
150 } 
151 
152 } 
153 
154 return (int)arrKey.GetSize(); 
155 } 
156 
157 BOOL CIni::DelAllSections() 
158 { 
159 int nSection; 
160 CStringArray arrSection; 
161 nSection=GetSections(arrSection); 
162 for(int i=0;i<nSection;i++) 
163 { 
164 if(DelSection(arrSection[i])) 
165 return GetLastError(); 
166 } 
167 return FALSE; 
168 } 
169 
170 
171 使用方法: 
172 CIni ini("c:\\a.ini"); 
173 int n; 
174 
175 /*获得值 
176 TRACE("%s",ini.GetValue("段1","键1")); 
177 */ 
178 
179 /*添加值 
180 ini.SetValue("自定义段","键1","值"); 
181 ini.SetValue("自定义段2","键1","值",false); 
182 */ 
183 
184 /*枚举全部段名 
185 CStringArray arrSection; 
186 n=ini.GetSections(arrSection); 
187 for(int i=0;i<n;i++) 
188 TRACE("%s\n",arrSection[i]); 
189 */ 
190 
191 /*枚举全部键名及值 
192 CStringArray arrKey,arrValue; 
193 n=ini.GetKeyValues(arrKey,arrValue,"段1"); 
194 for(int i=0;i<n;i++) 
195 TRACE("键:%s\n值:%s\n",arrKey[i],arrValue[i]); 
196 */ 
197 
198 /*删除键值 
199 ini.DelKey("段1","键1"); 
200 */ 
201 
202 /*删除段 
203 ini.DelSection("段1"); 
204 */ 
205 
206 /*删除全部 
207 ini.DelAllSections(); 
208 */

 

 

 ***************************************************************************
××××××××××第二、VC有关文件操作,参考资料2
***************************************************************************

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

1.头文件  File64.h

 

 1 class CFile64 : public CFile
 2 {
 3 public:
 4 
 5 // Attributes
 6  ULONGLONG GetPosition();
 7 
 8 
 9 // Overridables
10 
11  virtual ULONGLONG Seek(LONGLONG lOff, UINT nFrom);
12  virtual void SetLength(ULONGLONG dwNewLen);
13  ULONGLONG GetLength() ;
14 
15  virtual void LockRange(ULONGLONG dwPos, ULONGLONG dwCount);
16  virtual void UnlockRange(ULONGLONG dwPos, ULONGLONG dwCount);
17 
18 };

 

 

2.源文件  File64.cpp

 1 #include "stdafx.h"
 2 #include "file64.h"
 3 
 4 ////////////////////////////////////////////////////////////////////////////
 5 // CFile64 implementation
 6 
 7 
 8 ULONGLONG CFile64::Seek(LONGLONG lOff, UINT nFrom)
 9 {
10  ASSERT_VALID(this);
11  ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
12  ASSERT(nFrom == begin || nFrom == end || nFrom == current);
13  ASSERT(begin == FILE_BEGIN && end == FILE_END && current == FILE_CURRENT);
14 
15    LARGE_INTEGER liOff;
16 
17    liOff.QuadPart = lOff;
18  liOff.LowPart = ::SetFilePointer((HANDLE)m_hFile, liOff.LowPart, &liOff.HighPart,
19    (DWORD)nFrom);
20  if (liOff.LowPart  == (DWORD)-1)
21    if (::GetLastError() != NO_ERROR)
22      CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
23 
24  return liOff.QuadPart;
25 }
26 
27 ULONGLONG CFile64::GetPosition() 
28 {
29  ASSERT_VALID(this);
30  ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
31 
32    LARGE_INTEGER liPos;
33    liPos.QuadPart = 0;
34  liPos.LowPart = ::SetFilePointer((HANDLE)m_hFile, liPos.LowPart, &liPos.HighPart , FILE_CURRENT);
35  if (liPos.LowPart == (DWORD)-1)
36    if (::GetLastError() != NO_ERROR)
37      CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
38 
39  return liPos.QuadPart;
40 }
41 
42 void CFile64::LockRange(ULONGLONG dwPos, ULONGLONG dwCount)
43 {
44  ASSERT_VALID(this);
45  ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
46 
47    ULARGE_INTEGER liPos;
48    ULARGE_INTEGER liCount;
49 
50    liPos.QuadPart = dwPos;
51    liCount.QuadPart = dwCount;
52  if (!::LockFile((HANDLE)m_hFile, liPos.LowPart, liPos.HighPart, liCount.LowPart, 
53    liCount.HighPart))
54    {
55   CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
56    }
57 }
58 
59 void CFile64::UnlockRange(ULONGLONG dwPos, ULONGLONG dwCount)
60 {
61  ASSERT_VALID(this);
62  ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
63 
64    ULARGE_INTEGER liPos;
65    ULARGE_INTEGER liCount;
66 
67    liPos.QuadPart = dwPos;
68    liCount.QuadPart = dwCount;
69  if (!::UnlockFile((HANDLE)m_hFile, liPos.LowPart, liPos.HighPart, liCount.LowPart,
70    liCount.HighPart))
71    {
72   CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
73    }
74 }
75 
76 void CFile64::SetLength(ULONGLONG dwNewLen)
77 {
78  ASSERT_VALID(this);
79  ASSERT((HANDLE)m_hFile != INVALID_HANDLE_VALUE);
80 
81  Seek(dwNewLen, (UINT)begin);
82 
83  if (!::SetEndOfFile((HANDLE)m_hFile))
84   CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
85 }
86 
87 ULONGLONG CFile64::GetLength() 
88 {
89  ASSERT_VALID(this);
90 
91    ULARGE_INTEGER liSize;
92    liSize.LowPart = ::GetFileSize((HANDLE)m_hFile, &liSize.HighPart);
93    if (liSize.LowPart == (DWORD)-1)
94    if (::GetLastError() != NO_ERROR)
95    CFileException::ThrowOsError((LONG)::GetLastError(), m_strFileName);
96 
97  return liSize.QuadPart;
98 }
/////////////////////////////////////////////////////////////////////////////

 

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

 

整理自:http://www.cnblogs.com/lidabo/p/3470085.html

posted @ 2018-07-28 03:38  vranger  Views(922)  Comments(0Edit  收藏  举报