ini文件(即Initialization file),這種類型的文件中通常存放的是一個(gè)程序的初始化信息。ini文件由若干個(gè)節(jié)(Section)組成,每個(gè)Section由若干鍵(Key)組成,每個(gè)Key可以賦相應(yīng)的值。讀寫ini文件實(shí)際上就是讀寫某個(gè)的Section中相應(yīng)的Key的值,而這只要借助幾個(gè)函數(shù)即可完成。
一、向ini文件中寫入信息的函數(shù)
1. 把信息寫入系統(tǒng)的win.ini文件
BOOL WriteProfileString(
LPCTSTR lpAppName, // 節(jié)的名字,是一個(gè)以0結(jié)束的字符串
LPCTSTR lpKeyName, // 鍵的名字,是一個(gè)以0結(jié)束的字符串。若為NULL,則刪除整個(gè)節(jié)
LPCTSTR lpString // 鍵的值,是一個(gè)以0結(jié)束的字符串。若為NULL,則刪除對(duì)應(yīng)的鍵
)
2. 把信息寫入自己定義的.ini文件
BOOL WritePrivateProfileString(
LPCTSTR lpAppName, // 同上
LPCTSTR lpKeyName, // 同上
LPCTSTR lpString, // 同上
LPCTSTR lpFileName // 要寫入的文件的文件名。若該ini文件與程序在同一個(gè)目錄下,也可使用相對(duì)
//路徑,否則需要給出絕度路徑。
)
如:
::WriteProfileString("Test","id","xym");
//在win.ini中創(chuàng)建一個(gè)Test節(jié),并在該節(jié)中創(chuàng)建一個(gè)鍵id,其值為xym
::WritePrivateProfileString("Test","id","xym","d:\\vc\\Ex1\\ex1.ini");
//在Ex1目錄下的ex1.ini中創(chuàng)建一個(gè)Test節(jié),并在該節(jié)中創(chuàng)建一個(gè)鍵id,其值為xym
//若Ex1.ini文件與讀寫該文件的程序在同一個(gè)目錄下,則上面語(yǔ)句也可寫為:
::WritePrivateProfileString("Test","id","xym",".\\ex1.ini");
需要注意的是,C系列的語(yǔ)言中,轉(zhuǎn)義字符'\\'表示反斜線'\'。另外,當(dāng)使用相對(duì)路徑時(shí),\\前的.號(hào)不能丟掉了。
二、從ini文件中讀取數(shù)據(jù)的函數(shù)
1、從系統(tǒng)的win.ini文件中讀取信息
(1) 讀取字符串
DWORD GetProfileString(
LPCTSTR lpAppName, // 節(jié)名
LPCTSTR lpKeyName, // 鍵名,讀取該鍵的值
LPCTSTR lpDefault, // 若指定的鍵不存在,該值作為讀取的默認(rèn)值
LPTSTR lpReturnedString, // 一個(gè)指向緩沖區(qū)的指針,接收讀取的字符串
DWORD nSize // 指定lpReturnedString指向的緩沖區(qū)的大小
)
如:
CString str;
::GetProfileString("Test","id","Error",str.GetBuffer(20),20);
(2) 讀取整數(shù)
UINT GetProfileInt(
LPCTSTR lpAppName, // 同上
LPCTSTR lpKeyName, // 同上
INT nDefault // 若指定的鍵名不存在,該值作為讀取的默認(rèn)值
)
如使用以下語(yǔ)句寫入了年齡信息:
::WriteProfileString("Test","age","25");
//在win.ini中創(chuàng)建一個(gè)Test節(jié),并在該節(jié)中創(chuàng)建一個(gè)鍵age,其值為25
則可用以下語(yǔ)句讀取age鍵的值:
int age;
age=::GetProfileInt("Test","age",0);
2、從自己的ini文件中讀取信息
(1) 讀取字符串
DWORD GetPrivateProfileString(
LPCTSTR lpAppName, // 同1(1)
LPCTSTR lpKeyName, // 同1(1)
LPCTSTR lpDefault, // 同1(1)
LPTSTR lpReturnedString, // 同1(1)
DWORD nSize, // 同1(1)
LPCTSTR lpFileName // 讀取信息的文件名。若該ini文件與程序在同一個(gè)目錄下,也可使用相
//對(duì)路徑,否則需要給出絕度路徑。
)
如:
CString str;
::GetPrivateProfileString("Test","id","Error",str.GetBuffer(20),20,".\\ex1.ini");
或:
::GetPrivateProfileString("Test","id","Error",str.GetBuffer(20),20,"d:\\vc\\Ex1\\ex1.ini");
(2) 讀取整數(shù)
UINT GetPrivateProfileInt(
LPCTSTR lpAppName, // 同上
LPCTSTR lpKeyName, // 同上
INT nDefault, // 若指定的鍵名不存在,該值作為讀取的默認(rèn)值
LPCTSTR lpFileName // 同上
)
如使用以下語(yǔ)句寫入了年齡信息:
::WritePrivateProfileString("Test","age","25",".\\ex1.ini");
//在ex1.ini中創(chuàng)建一個(gè)Test節(jié),并在該節(jié)中創(chuàng)建一個(gè)鍵age,其值為25
則可用以下語(yǔ)句讀取age鍵的值:
int age;
age=::GetPrivateProfileInt("Test","age",0,".\\ex1.ini");
三、 刪除鍵值或節(jié)
回顧一下WriteProfileString函數(shù)的說(shuō)明
BOOL WriteProfileString(
LPCTSTR lpAppName, // 節(jié)的名字,是一個(gè)以0結(jié)束的字符串
LPCTSTR lpKeyName, // 鍵的名字,是一個(gè)以0結(jié)束的字符串。若為NULL,則刪除整個(gè)節(jié)
LPCTSTR lpString // 鍵的值,是一個(gè)以0結(jié)束的字符串。若為NULL,則刪除對(duì)應(yīng)的鍵
)
由此可見(jiàn),要?jiǎng)h除某個(gè)節(jié),只需要將WriteProfileString第二個(gè)參數(shù)設(shè)為NULL即可。而要?jiǎng)h除某個(gè)鍵,則只需要將該函數(shù)的第三個(gè)參數(shù)設(shè)為NULL即可。這是刪除系統(tǒng)的win.ini中的節(jié)或鍵,類似的,要?jiǎng)h除自己定義的ini文件中的節(jié)或鍵,也可做相同的操作。
如:
::WriteProfileString("Test",NULL,NULL); //刪除win.ini中的Test節(jié)
::WriteProfileString("Test","id",NULL); //刪除win.ini中的id鍵
::WritePrivateProfileString("Test",NULL,NULL,".\\ex1.ini"); //刪除ex1.ini中的Test節(jié)
::WritePrivateProfileString("Test","id",NULL,".\\ex1.ini"); //刪除ex1.ini中的id鍵
四、如何判斷一個(gè)ini文件中有多少個(gè)節(jié)
要判斷一個(gè)ini文件中有多少個(gè)節(jié),最簡(jiǎn)單的辦法就是將所有的節(jié)名都找出來(lái),然后統(tǒng)計(jì)節(jié)名的個(gè)數(shù)。而要將所有的節(jié)名找出來(lái),使用GetPrivateProfileSectionNames函數(shù)就可以了,其原型如下:
DWORD GetPrivateProfileSectionNames(
LPTSTR lpszReturnBuffer, // 指向一個(gè)緩沖區(qū),用來(lái)保存返回的所有節(jié)名
DWORD nSize, // 參數(shù)lpszReturnBuffer的大小
LPCTSTR lpFileName // 文件名,若該ini文件與程序在同一個(gè)目錄下,
//也可使用相對(duì)路徑,否則需要給出絕度路徑
)
下面的是用來(lái)統(tǒng)計(jì)一個(gè)ini文件中共有多少個(gè)節(jié)的函數(shù),當(dāng)然,如果需要同時(shí)找到每個(gè)節(jié)中的各個(gè)鍵及其值,根據(jù)找到節(jié)名就可以很容易的得到了。
/*統(tǒng)計(jì)共有多少個(gè)節(jié)
節(jié)名的分離方法:若chSectionNames數(shù)組的第一字符是'\0'字符,則表明
有0個(gè)節(jié)。否則,從chSectionNames數(shù)組的第一個(gè)字符開(kāi)始,順序往后找,
直到找到一個(gè)'\0'字符,若該字符的后繼字符不是 '\0'字符,則表明前
面的字符組成一個(gè)節(jié)名。若連續(xù)找到兩個(gè)'\0'字符,則統(tǒng)計(jì)結(jié)束*/
int CTestDlg::CalcCount(void)
{
TCHAR chSectionNames[2048]={0}; //所有節(jié)名組成的字符數(shù)組
char * pSectionName; //保存找到的某個(gè)節(jié)名字符串的首地址
int i; //i指向數(shù)組chSectionNames的某個(gè)位置,從0開(kāi)始,順序后移
int j=0; //j用來(lái)保存下一個(gè)節(jié)名字符串的首地址相對(duì)于當(dāng)前i的位置偏移量
int count=0; //統(tǒng)計(jì)節(jié)的個(gè)數(shù)
//CString name;
//char id[20];
::GetPrivateProfileSectionNames(chSectionNames,2048,".\\ex1.ini");
for(i=0;i<2048;i++,j++)
{
if(chSectionNames[0]=='\0')
break; //如果第一個(gè)字符就是0,則說(shuō)明ini中一個(gè)節(jié)也沒(méi)有
if(chSectionNames[i]=='\0')
{
pSectionName=&chSectionNames[i-j]; //找到一個(gè)0,則說(shuō)明從這個(gè)字符往前,減掉j個(gè)偏移量,
//就是一個(gè)節(jié)名的首地址
j=-1; //找到一個(gè)節(jié)名后,j的值要還原,以統(tǒng)計(jì)下一個(gè)節(jié)名地址的偏移量
//賦成-1是因?yàn)楣?jié)名字符串的最后一個(gè)字符0是終止符,不能作為節(jié)名
//的一部分
/*::GetPrivateProfileString(pSectionName,"id","Error",id,20,".\\ex1.ini");
name.Format("%s",id);*/
//在獲取節(jié)名的時(shí)候可以獲取該節(jié)中鍵的值,前提是我們知道該節(jié)中有哪些鍵。
AfxMessageBox(pSectionName); //把找到的顯示出來(lái)
if(chSectionNames[i+1]==0)
{
break; //當(dāng)兩個(gè)相鄰的字符都是0時(shí),則所有的節(jié)名都已找到,循環(huán)終止
}
}
}
return count;
}