2024年2月2日发(作者:)

0

在Visual C++ 6.0下对Windows配置设置文件的存取访问方法的详细介绍。0 一、 引言0

不论是对于程序开发人员还是软件应用人员,一定不会对扩展名为"ini"的文件感到陌生,不仅Windows操作系统将大名鼎鼎的作为记录当前系统状态,并根据其记录内容对系统进行配置的一种便捷的方法,而且众多的应用软件也广泛地使用该类型的配置文件来对软件进行记录、配置。本文就针对配置设置文件的使用展开讨论,以期能为软件状态的记录与设置寻求一种方便简洁的实现方法。 二、 配置设置文件概述0

0

配置设置文件是Windows操作系统下的一种特殊化的ASCII文件,以"ini"为文件扩展名。该文件也被称做是初始化文件(initialization file)和概要文件(profile),通常应用程序可以拥有自己的配置设置文件来存储自己的状态信息,一般来说私有的配置设置文件比较小,可以减少程序在初始化时读取配置文件时的信息量,从而可以提高程序的启动速度、提高应用程序和系统的性能。但如待存取的信息涉及到Windows系统环境或是其他的应用程序时才必须在Windows系统的配置文件中记录并在访问的同时发送出消息WM_WININICHANGE给所有的顶层窗口,通知其他的程序系统的配置文件已做了更改。但由于中不仅记录了系统的有关信息,也存储着许多其他应用软件的一

些配置数据,所以访问的数据量要远比私有的配置文件大的多。0

配置文件里的信息之所以能为系统和众多的软件所读取并识别,是由于其内部对数据的存取采用了预先约定好的"项-值对(entry-value

pairs)"存储结构来对待存取的数据进行分门别类地进行条理清晰的存储。我们可以打开系统目录下的文件:[windows]

load=

run=

NullPort=None0

0

[Desktop]

WallpaperStyle=2

Pattern=(无)0

[intl]

s2359=PM

iCountry=86

……0

可见,配置文件把信息分成若干"节",节标题放在方括号中,如[Desktop]就是Desktop节,在一个节内包含了一些与之相关相近的"项",并通过等号对其进行赋值。一般的形式如下所示:[SECTION]

ENTRY=VALUE0

0

其中VALUE值可以有两种类型:数值型和字符串型,而且Windows操作系统专门为此提供了6个API函数来对配置设置文件进行读、写:0

GetPrivateProfileInt() 从私有初始化文件获取整型数值

GetPrivateProfileString() 从私有初始化文件获取字符串型值

GetProfileInt 从 获取整数值

GetProfileString 从 获取字符串值

WritePrivateProfileString 写字符串到私有初始化文件

WriteProfileString 写字符串到0

在这里需要指出:当向配置文件存储信息时,不论是数据还是字符串都要先转换成字符串,然后再行存储。

0

利用GetPrivateProfileString读取配置文件(.ini)0

配置文件中经常用到ini文件,在VC中其函数分别为:0

写入.ini文件:bool WritePrivateProfileString(LPCTSTR

lpAppName,LPCTSTR lpKeyName,LPCTSTR lpString,LPCTSTR

lpFileName);0

读取.ini文件:DWORD GetPrivateProfileString(LPCTSTR

lpAppName,LPCTSTR lpKeyName,LPCTSTR lpDefaut,LPSTR

lpReturnedString,DWORD nSize,LPCTSTR lpFileName);0

读取整形值:UINT GetPrivateProfileInt(LPCTSTR

lpAppName,LPCTSTR lpKeyName,INT nDefault,LPCTSTR

lpFileName);0

其中个参数的意思:0

LPCTSTR lpAppName ------- INI文件中的一个字段名0

LPCTSTR lpKeyName -------- lpAppName 下的一个键名,也就是里面具体的变量名0

LPCTSTR lpString ---------是键值,也就是变量的值, 必须为LPCTSTR或CString类型0

LPCTSTR lpFileName --------完整的INI文件路径名0

LPCTSTR lpDefaut ----------如果没有其前两个参数值,则将此值赋给变量0

LPSTR lpReturnedString --------接收INI文件中的值的CString对象,即接收缓冲区0

DWORD nSize ------接收缓冲区的大小例子:0

0

CString StrName,Strtemp;int nAge;00

StrName = "jacky";nAge = 13;0

0

WritePrivateProfileString("Student","Name",StrName,"c:");0

结果:(INI文件中显示如下:)0

[Student]0

Name=jacky读取:0

0

CString SName;0

GetPrivateProfileString("Student","Name","DefaultName",fer(MAX_LENGTH),MAX_LENGTH,"c:");0

结果:SName = "jacky";这里需要注意点就是用完GetBuffer函数后一定要释放(用eBuffer()函数),不然后面再用到SName的其他子函数就会失灵。读整数比较简单,如下0

0

int Result = GetPrivateProfileInt("Student","nAge",0,"c:")返回值即为所读取的结果!0

在GetPrivateProfileString最后一个参数是配置文件路径的参数,此路径只能是绝对路径,不能是相对路径,但现在我需要是我的exe文件能和我的配置文件在一起。因此我使用了GetCurrentDirectory函数。0

原代码如下:0

CString server_ip;

CString des="";

::GetCurrentDirectory(MAX_PATHLENGTH,fer(MAX_PATHLENGTH));

eBuffer();

des+="";

GetPrivateProfileString("PhoneDemo","Server_IP","",server_

fferSetLength(15),15,des);

server_eBuffer();0

注意:在这里使用CString变量时,在使用完GetBuffer后,紧接着一定要使用ReleaseBuffer()函数,才可以进行其他的诸如字符串+操作0

vc 读写 init文件(软件配置)0

VC中用函数读写ini文件的方法

ini文件(即Initialization file),这种类型的文件中通常存放的是一个程序的初始化信息。ini文件由若干个节(Section)组成,每个Section由若干键(Key)组成,每个Key可以赋相应的值。读写ini文件实际上就是读写某个的Section中相应的Key的值,而这只要借助几个函数即可完成。

一、向ini文件中写入信息的函数

1. 把信息写入系统的文件

BOOL WriteProfileString(

LPCTSTR lpAppName, // 节的名字,是一个以0结束的字符串

LPCTSTR lpKeyName, // 键的名字,是一个以0结束的字符串。若为NULL,则删除整个节

LPCTSTR lpString // 键的值,是一个以0结束的字符串。若为NULL,则删除对应的键

)

0

2. 把信息写入自己定义的.ini文件

BOOL WritePrivateProfileString(

LPCTSTR lpAppName, // 同上

LPCTSTR lpKeyName, // 同上

LPCTSTR lpString, // 同上

LPCTSTR lpFileName // 要写入的文件的文件名。若该ini文件与程序在同一个目录下,也可使用相对

//路径,否则需要给出绝度路径。

)

0

如:

::WriteProfileString("Test","id","xym");

//在中创建一个Test节,并在该节中创建一个键id,其值为xym

0::WritePrivateProfileString("Test","id","xym","d:");

//在Ex1目录下的中创建一个Test节,并在该节中创建一个键id,其值为xym

0//若文件与读写该文件的程序在同一个目录下,则上面语句也可写为:

::WritePrivateProfileString("Test","id","xym",".");

0需要注意的是,C系列的语言中,转义字符''表示反斜线''。另外,当使用相对路径时,前的.号不能丢掉了。

0二、从ini文件中读取数据的函数

1、从系统的文件中读取信息

(1) 读取字符串

DWORD GetProfileString(

LPCTSTR lpAppName, // 节名

LPCTSTR lpKeyName, // 键名,读取该键的值

LPCTSTR lpDefault, // 若指定的键不存在,该值作为读取的默认值

LPTSTR lpReturnedString, // 一个指向缓冲区的指针,接收读取的字符串

DWORD nSize // 指定lpReturnedString指向的缓冲区的大小

)

0如:

CString str;

::GetProfileString("Test","id","Error",fer(20),20);0

(2) 读取整数

UINT GetProfileInt(

LPCTSTR lpAppName, // 同上

LPCTSTR lpKeyName, // 同上

INT nDefault // 若指定的键名不存在,该值作为读取的默认值

)

0如使用以下语句写入了年龄信息:

::WriteProfileString("Test","age","25");

//在中创建一个Test节,并在该节中创建一个键age,其值为25

0则可用以下语句读取age键的值:

int age;

age=::GetProfileInt("Test","age",0);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文件与程序在同一个目录下,也可使用相

//对路径,否则需要给出绝度路径。

)

0如:

CString str;

::GetPrivateProfileString("Test","id","Error",fer(20),20,".");

或:

::GetPrivateProfileString("Test","id","Error",fer(20),20,"d:");

0(2) 读取整数0

UINT GetPrivateProfileInt(

LPCTSTR lpAppName, // 同上

LPCTSTR lpKeyName, // 同上

INT nDefault, // 若指定的键名不存在,该值作为读取的默认值

LPCTSTR lpFileName // 同上

)

0如使用以下语句写入了年龄信息:

::WritePrivateProfileString("Test","age","25",".");

//在中创建一个Test节,并在该节中创建一个键age,其值为25

0则可用以下语句读取age键的值:

int age;

age=::GetPrivateProfileInt("Test","age",0,".");三、 删除键值或节00

回顾一下WriteProfileString函数的说明

BOOL WriteProfileString(

LPCTSTR lpAppName, // 节的名字,是一个以0结束的字符串

LPCTSTR lpKeyName, // 键的名字,是一个以0结束的字符串。若为NULL,则删除整个节

LPCTSTR lpString // 键的值,是一个以0结束的字符串。若为NULL,

则删除对应的键

)0

由此可见,要删除某个节,只需要将WriteProfileString第二个参数设为NULL即可。而要删除某个键,则只需要将该函数的第三个参数设为NULL即可。这是删除系统的中的节或键,类似的,要删除自己定义的ini文件中的节或键,也可做相同的操作。

如:

::WriteProfileString("Test",NULL,NULL); //删除中的Test节

::WriteProfileString("Test","id",NULL); //删除中的id键

0::WritePrivateProfileString("Test",NULL,NULL,"."); //删除中的Test节

::WritePrivateProfileString("Test","id",NULL,"."); //删除中的id键

0四、如何判断一个ini文件中有多少个节

要判断一个ini文件中有多少个节,最简单的办法就是将所有的节名都找出来,然后统计节名的个数。而要将所有的节名找出来,使用GetPrivateProfileSectionNames函数就可以了,其原型如下:

DWORD GetPrivateProfileSectionNames(

LPTSTR lpszReturnBuffer, // 指向一个缓冲区,用来保存返回的所有节名

DWORD nSize, // 参数lpszReturnBuffer的大小

LPCTSTR lpFileName // 文件名,若该ini文件与程序在同一个目录下,

0 //也可使用相对路径,否则需要给出绝度路径

)

0下面的是用来统计一个ini文件中共有多少个节的函数,当然,如果需要同时找到每个节中的各个键及其值,根据找到节名就可以很容易的得到了。

0

/*统计共有多少个节

节名的分离方法:若chSectionNames数组的第一字符是'0'字符,则表明

有0个节。否则,从chSectionNames数组的第一个字符开始,顺序往后找,

直到找到一个'0'字符,若该字符的后继字符不是 '0'字符,则表明前

面的字符组成一个节名。若连续找到两个'0'字符,则统计结束*/

0

int CTestDlg::CalcCount(void)

{

TCHAR chSectionNames[2048]={0}; //所有节名组成的字符数组

char * pSectionName; //保存找到的某个节名字符串的首地址

int i; //i指向数组chSectionNames的某个位置,从0开始,顺序后移

int j=0; //j用来保存下一个节名字符串的首地址相对于当前i的位置偏移量

int count=0; //统计节的个数

0//CString name;

//char id[20];

::GetPrivateProfileSectionNames(chSectionNames,2048,".");

for(i=0;i<2048;i++,j++)

{

if(chSectionNames[0]=='0')

break; //如果第一个字符就是0,则说明ini中一个节也没有

if(chSectionNames[i]=='0')

{

pSectionName=&chSectionNames[i-j]; //找到一个0,则说明从这个字符往前,减掉j个偏移量,

//就是一个节名的首地址

0 j=-1; //找到一个节名后,j的值要还原,以统计下一个节名地址的偏移量

//赋成-1是因为节名字符串的最后一个字符0是终止符,不能作为节名

0 //的一部分

/*::GetPrivateProfileString(pSectionName,"id","Error",id,20,".");

("%s",id);*/

//在获取节名的时候可以获取该节中键的值,前提是我们知道该节中有哪些键。

AfxMessageBox(pSectionName); //把找到的显示出来

0 if(chSectionNames[i+1]==0)

{

break; //当两个相邻的字符都是0时,则所有的节名都已找到,循环终止

}

}

}

0return count;

}

0

C技巧:VC读取和分析格式化文本配置文件(1)

0

大多数的应用程序都有需要配置的参数,配置参数的保存有多种方法,比如采用数据库保存、二进制文件保存、格式化的文本文件保存,各有优缺点,对于一般小的安全性要求不高应用程序,推荐采用格式化文本文件保存方式,这样可以节约编辑修改的界面编程,使用记事本程序编辑即可。

0

早期的windows就是采用的INI格式文本文件。现在很多应用程序包括大型的软件也还在采用格式化文本文件保存配置参数,比如Bentely MicroStation GIS平台GEOGRAPHICS8.0。0

本文给出的就是利用VC来读取和分析配置文本文件,这是笔者这消防监控系统远程终端软件使用的方法。同时给出一个C++类来封装配置文本文件的读取分析。0

0 下面是我采用的配置文件的例子,包含单个参数定义,表格参数定义,注释: // 协议名称: TL // 工程: SQ0

0

// 配置人: JY0 #DEFINE_BEGIN //以下为常量定义 //通信参数00

@COM_PORT .COM1 //COM口0

@COM_BAUDRATE 1200 //波特率0

@COM_BITSIZE 8 //数据位 number of bits/byte, 4-8 @COM_STOPBIT 0 // 0,1,2 对应 1, 1.5, 200

@COM_PARITY 0 //0-4对应no,odd,even,mark,space @STATIONNO 0 //终端号00

@TCPIP_SERVERIP 192.168.0.10 //服务器IP地址 @TCPIP_PORTNO 3024 //端口 #DEFINE_END00

0

#TABLE_BEGIN //表格 //编号 识别字符串 //例子如下00

0

0 0 Ion Detector, ALU1, L1S43 //测量台2#离子烟感探测器 报警 1 Thermal Det, ALU1, L1S39 //值班室温感探测器报警 #TABLE_END 语法解释:0

0

0

0 // 单行注释0 #DEFINE_BEGIN 单个参数定义的开始标记 #DEFINE_END 单个参数定义的结束标记 #TABLE_BEGIN 表格参数开始标记 #TABLE_END 表格参数结束标记 单个参数的配置:

0

0

0

0

0 每个参数一行:以空格、TAB制表符作为分隔,注释可选,每行结束后有Enter键换行。

表格参数配置:第一列为数字编号,作为ID用,第一列和第二列之间采用空格、TAB制表符作为分隔,其他列之间采用’,’逗号、空格、TAB制表符分隔,因此从第二列起,字符串中间夹的空格是有效字符。比如上面例子中的第一行解释如下:0

0 "0","Ion Detector","ALU1","L1S43"

同样,表格参数也是每行有Enter键换行。0 下面给出一个C++类来完成这个格式的文本配置读取和分析。 规模:00

文本文件的大小:<48kbytes 单个参数定义个数:不限00

表格参数定义: 100行 X 4列0 这些规模大小可以修改下面的宏定义来满足你的需要 #define TABLE_MAXROW 10000

#define TABLE_MAXCOL 5 //含索引ID列 typedef struct{ int nID;00

0

char s[TABLE_MAXCOL-1][64];0 }TAG_TABLEREC,*LPTABLEREC; class CDataTxt {00

0

public:0

CDataTxt();0

~CDataTxt();0 CString m_szPath;//配置文件全路经名 BOOL Read();//读入文件00