关于C#操作INI文件的总结

INI文件其实是一种具有特定结构的文本文件,它的构成分为三部分,结构如下: [Section1] key 1 = value2 key 1 = value2 …… [Section2] key 1 = value1 key 2 = value2 …… 文件由若干个段落(section)组成,每个段落又分成若干个键(key)和值(value)。Windows系统自带的Win32的API函数GetPrivateProfileString()和WritePrivateProfileString()分别实现了对INI文件的读写操作,他们位于kernel32.dll下。 但是令人遗憾的是C#所使用的.NET框架下的公共类库并没有提供直接操作INI文件的类,所以唯一比较理想的方法就是调用API函数。 然后,.Net框架下的类库是基于托管代码的,而API函数是基于非托管代码的,(在运行库的控制下执行的代码称作托管代码。相反,在运行库之外运行的代码称作非托管代码。)如何实现托管代码与非托管代码之间的操作呢?.Net框架的System.Runtime.InteropServices命名空间下提供各种各样支持COM interop及平台调用服务的成员,其中最重要的属性之一DllImportAttribute可以用来定义用于访问非托管API的平台调用方法,它提供了对从非托管DLL导出的函数进行调用所必需的信息。下面就来看一下如何实现C#与API函数的互操作。 读操作: [DllImport("kernel32")] private static extern int GetPrivateProfileString(string section, string key, string defVal, StringBuilder retVal, int size, string filePath); section:要读取的段落名 key: 要读取的键 defVal: 读取异常的情况下的缺省值 retVal: key所对应的值,如果该key不存在则返回空值 size: 值允许的大小 filePath: INI文件的完整路径和文件名 写操作: [DllImport("kernel32")] private static extern long WritePrivateProfileString(string section, string key, string val, string filePath); section: 要写入的段落名 key: 要写入的键,如果该key存在则覆盖写入 val: key所对应的值 filePath: INI文件的完整路径和文件名 这样,在就可以使用对他们的调用,用常规的方式定义一个名为IniFile类: 1 using System; 2 using System.Runtime.InteropServices; 3 using System.Text; 4 5 namespace IPVOD.Hotel.Remoting 6 { 7 /// <summary> 8 /// INI文件的操作类 9 /// </summary> 10 public class IniFile 11 { 12 public string Path; 13 14 public IniFile(string path) 15 { 16 this.Path = path; 17 } 18 19 声明读写INI文件的API函数 29 30 /// <summary> 31 /// 写INI文件 32 /// </summary> 33 /// <param name="section">段落</param> 34 /// <param name="key">键</param> 35 /// <param name="iValue">值</param> 36 public void IniWriteValue(string section, string key, string iValue) 37 { 38 WritePrivateProfileString(section, key, iValue, this.Path); 39 } 40 41 /// <summary> 42 /// 读取INI文件 43 /// </summary> 44 /// <param name="section">段落</param> 45 /// <param name="key">键</param> 46 /// <returns>返回的键值</returns> 47 public string IniReadValue(string section, string key) 48 { 49 StringBuilder temp = new StringBuilder(255); 50 51 int i = GetPrivateProfileString(section, key, "", temp, 255, this.Path); 52 return temp.ToString(); 53 } 54 55 /// <summary> 56 /// 读取INI文件 57 /// </summary> 58 /// <param name="Section">段,格式[]</param> 59 /// <param name="Key">键</param> 60 /// <returns>返回byte类型的section组或键值组</returns> 61 public byte[] IniReadValues(string section, string key) 62 { 63 byte[] temp = new byte[255]; 64 65 int i = GetPrivateProfileString(section, key, "", temp, 255, this.Path); 66 return temp; 67 } 68 } 69 } 70 注意:我增加了DLL导出的函数GetPrivateProfileString的重载,说明如下: [DllImport("kernel32")] private static extern int GetPrivateProfileString(string section, string key, string defVal, Byte[] retVal, int size, string filePath); section:要读取的段落名 key: 要读取的键 defVal: 读取异常的情况下的缺省值 retVal: 此参数类型不是string,而是Byte[]用于返回byte类型的section组或键值组。 size: 值允许的大小 filePath: INI文件的完整路径和文件名 下面看一下具体实例化IniFile类的操作: //path为ini文件的物理路径 IniFile ini = new IniFile(path); //读取ini文件的所有段落名 byte[] allSection = ini.IniReadValues(null, null); 通过如下方式转换byte[]类型为string[]数组类型 string[] sectionList; ASCIIEncoding ascii = new ASCIIEncoding(); //获取自定义设置section中的所有key,byte[]类型 sectionByte = ini.IniReadValues("personal", null); //编码所有key的string类型 sections = ascii.GetString(sectionByte); //获取key的数组 sectionList = sections.Split(new char[1]{'