C# 如何实现完整的INI文件读写类

https://www.cnblogs.com/ps3online/archive/2020/02/28/INIFile.htmlhtml


 

做者: 魔法软糖node

日期: 2020-02-27windows

引言ide

*************************************ui

.ini 文件是Initialization File的缩写,即配置文件 。windows的系统配置文件所采用的存储格式。编码

它具备方便易用的特色,和注册表的键值有着相似的功能,各种应用程序也常用INI保存各类配置和选项。spa

在简单只须要读取的场合,调用WINDOWS API就行,但在复杂的须要可视化编辑时,就须要创建本身的处理类了。3d

 

 

 

如何实现本身的INI类调试

首先咱们须要了解code

◇  INI的格式

 

 

◇  典型INI文件

;项目注释
[.ShellClassInfo]
InfoTip=有图标的文件夹
;图标资源
IconResource="C:\Windows\system32\SHELL32.dll",4
#文件夹视图
[ViewState]
Mode=
Vid=
FolderType=General
#尾部注释









 一个典型INI文件由节、注释和节下面的项组成,而项为键=值的形式。

INI文件的注释符号有两种,规范为;分号,实际有些地方用#井号。

◇  保留注释

为了在修改INI文件时不丢失任何信息,因此须要保存INI文件中全部有效元素、包括注释甚至是无效行。

为了实现这个目的,将全部注释和无效行都归属于它以后的有效元素。

以上面的desktop.ini为例,

  • 第一行 ;项目注释归属于[.ShellClassInfo]
  • 第四行;图标资源归属于IconResource=
  • #文件夹视图归属于[ViewState]
  • 最后的#尾部注释归属于整个INI文档的EndText

◇  INIItem类

表示INI文件中节点下面的项,它拥有三个属性:名称Name、值Value和注释Comment

 1     /// <summary>
 2     /// 拥有名称、值和注释  3     /// </summary>
 4     public class INIItem {  5         /// <summary>
 6         /// 实例化INIItem。指定名称、值和注释。  7         /// </summary>
 8         /// <param name="vName"></param>
 9         /// <param name="vValue"></param>
10         /// <param name="vComment"></param>
11         public INIItem(string vName, string vValue, string vComment = "") { 12             Name = vName; 13             Value = vValue; 14             Comment = vComment; 15  } 16         /// <summary>
17         /// 项名称。例如 Color = 202,104,0 中的 Color 18         /// </summary>
19         public string Name { get; set; } 20         /// <summary>
21         /// 值内容。例如 Color = 202,104,0 中的 202,104,0 22         /// </summary>
23         public string Value { get; set; } 24         /// <summary>
25         /// 位于前面的全部注释行。通常以 ; 开头 26         /// </summary>
27         public string Comment { get; set; } 28         /// <summary>
29         /// 返回 INIItem 的文本形式。〈<see cref="string"/>30         /// <para>Name=Value</para>
31         /// </summary>
32         /// <returns>〈string〉返回 INIItem 的文本形式。</returns>
33         public override string ToString() { 34             return Name + INI.U等号 + Value; 35  } 36     }

◇  ININode类

表示INI文件中的一个节点,它拥有项列表List{Of INIItem}、名称Name和注释Comment。

 1     /// <summary>
 2     /// 表示INI文件的一个节点,它拥有一个项目列表,还拥有名称和注释  3     /// <para></para>
 4     /// </summary>
 5     public class ININode {  6         /// <summary>
 7         /// 实例化ININode。指定初始的名称和注释。  8         /// </summary>
 9         /// <param name="vName"></param>
10         /// <param name="vComment"></param>
11         public ININode(string vName, string vComment) { Name = vName; Comment = vComment; Items = new List<INIItem>(); } 12         /// <summary>
13         /// 节点名称。例如 [Config] 14         /// </summary>
15         public string Name { get; set; } 16         /// <summary>
17         /// 位于前面的全部注释行。通常以 ; 开头 18         /// </summary>
19         public string Comment { get; set; } 20         /// <summary>
21         /// 含有的项列表 22         /// </summary>
23         public List<INIItem> Items { get; set; } 24         /// <summary>
25         /// 向本节点添加新项。 26         /// </summary>
27         /// <param name="vName"></param>
28         /// <param name="vValue"></param>
29         /// <param name="vComment"></param>
30         /// <returns></returns>
31         public INIItem New(string vName, string vValue, string vComment = "") { 32             var k = new INIItem(vName, vValue, vComment); 33  Items.Add(k); 34             return k; 35  } 36         /// <summary>
37         /// 返回 ININode的文本形式。〈<see cref="string"/>38         /// <para>[Name]</para>
39         /// </summary>
40         /// <returns>〈string〉返回 ININode 的文本形式。</returns>
41         public override string ToString() { 42             return INI.U左括号 + Name + INI.U右括号; 43  } 44     }

◇  INI类

它表示整个INI文件的所有内,拥有List{Of ININode}、EndText、FileName、StartLine等属性

 1     /// <summary>
 2     /// 表示INI文件。拥有读取和写入文件的方法。  3     /// <para>储存在 <see cref="List{ININode}"/>&lt;<see cref="ININode"/>&gt;</para>
 4     /// </summary>
 5     public class INI {  6         /// <summary>
 7         /// 实例化INI文件。  8         /// </summary>
 9         public INI() { } 10 
11         #region "↓全局常量"
12         /// <summary>注释的标准符号</summary>
13         public static string U注释 = ";"; 14         /// <summary>注释的标准符号2</summary>
15         public static string U注释2 = "#"; 16         /// <summary>节左括号的标准符号</summary>
17         public static string U左括号 = "["; 18         /// <summary>节右括号的标准符号</summary>
19         public static string U右括号 = "]"; 20         /// <summary>链接项和值的标准符号</summary>
21         public static string U等号 = "="; 22         /// <summary>读取或写入时忽略无心义的备注行(不包括注释)。</summary>
23         public static bool 忽略备注 = false; 24         /// <summary>读取的上个文件的有效行数(不包括注释)。</summary>
25         public static int 上次读取的有效行数 = 0; 26         #endregion
27 
28         /// <summary>
29         /// 全部节点 30         /// <para>每一个节点含有项、值和注释,当项名称为空字符串时,整条语句视为注释</para>
31         /// </summary>
32         public List<ININode> Nodes { get; set; } = new List<ININode>(); 33         /// <summary>
34         /// 附加在INI文件后无心义的文本 35         /// </summary>
36         public string EndText { get; set; } = ""; 37         /// <summary>
38         /// 附加在INI文件第一行的做者信息等文本 39         /// <para>其中的换行符将被替换为两个空格</para>
40         /// </summary>
41         public string StartLine { get; set; } = ""; 42         /// <summary>
43         /// 读取INI时得到的FileName。 44         /// <para>写入文档时可使用这个名字,也能够不使用这个名字。</para>
45         /// </summary>
46         public string FileName { get; set; } = ""; 47         /// <summary>
48         /// 向本INI文件添加新节点。 49         /// </summary>
50         /// <param name="vName"></param>
51         /// <param name="vComment"></param>
52         /// <returns></returns>
53         public ININode New(string vName, string vComment = "") { 54             var k = new ININode(vName, vComment); 55  Nodes.Add(k); 56             return k; 57  } 58     }

如何写入INI文件

  1. 首先遍历每一个节点,写入节点的注释节点名称(套个括号)
  2. 而后遍历每一个节点下面的,写入项的注释项的名称=值
  3. 写入尾部注释

如下是写入代码

 1         #region "写入文件"
 2 
 3         /// <summary>将文档写入指定路径  4         /// </summary>
 5         /// <param name="path">指定路径</param>
 6         public bool 写入文档(string path, Encoding encoding = null) {  7             try {  8                 if (encoding == null) { encoding = Encoding.Default; }  9                 using (StreamWriter SW = new StreamWriter(path)) { 10  SW.Write(ToString()); 11  } 12             } catch (Exception) { 13                 return false; 14  } 15             return true; 16  } 17         /// <summary>
18         /// 将INI文档转化为文本格式,会生成整个文档。 19         /// <para>注意:较大的文档可能会耗费大量时间</para>
20         /// </summary>
21         /// <returns></returns>
22         public override string ToString() { 23             StringBuilder sb = new StringBuilder(); 24             if (StartLine.Length > 0) { sb.AppendLine(StartLine.Replace("\r\n", "  ")); } 25             for (int i = 0; i < Nodes.Count; i++) { 26                 var node = Nodes[i]; 27                 if (忽略备注 == false) { sb.Append(node.Comment); } 28  sb.AppendLine(node.ToString()); 29                 for (int j = 0; j < node.Items.Count; j++) { 30                     var item = node.Items[j]; 31                     if (忽略备注 == false) { sb.Append(item.Comment); } 32  sb.AppendLine(item.ToString()); 33  } 34  } 35             if (EndText.Length > 0) { sb.AppendLine(EndText); } 36             return sb.ToString(); 37  } 38 
39         #endregion

 

 

如何读取INI文件

读取一般比写入复杂。软糖的代码也是逐行检查,屡次调试才完成。

流程以下:

  1. 首先定义一些局部变量来记录当前分析的节、项、已经累积的备注、是否为有效行
  2. 逐行读取,首先判断是否开头为;#,若是是,添加到备注,加回车符,设为有效行。
  3. 判断开头是否为[,若是是则做为节来读取,进一步分析,若是[A]这种形式,设置当前节,设为有效行,若是[B缺乏反括号,进行下一步流程,尚没法判断是[B=K这种项仍是纯粹无心义的无效行。
  4. 判断是否含有=,若是是则做为项来读取
  5. 若是未标记为有效行,统统加入备注
  6. 若是读彻底文,备注不为空,则加入到INI.EndText中做为结尾注释。

代码

#region "读取文件"
        /// <summary>
        /// 从指定路径和字符编码的文件中读取文档内容,以今生成本文档。 /// </summary>
        /// <param name="路径">完整的路径字符串</param>
        /// <param name="encoding">编码格式:默认自动识别。(对于无bom可能识别错误)</param>
        public bool 读取文档(string 路径, Encoding encoding = null) { if (File.Exists(路径) == false) { return false; } try { if (encoding == null) { encoding = TXT.GetFileEncodeType(路径); } using (StreamReader SR = new StreamReader(路径, encoding)) { bool 返回结果 = 读取文档(new StringReader(SR.ReadToEnd())); SR.Close(); return 返回结果; } } catch (Exception) { return false; } } /// <summary>
        ///<see cref="StringReader"/> 中读取文档内容,以今生成本文档。 /// </summary>  
        /// <param name="MyStringReader">StringReader,能够由string或StreamReader.ReadToEnd()来生成。</param>
        /// <returns>〈bool〉返回是否读取成功。</returns>
        public bool 读取文档(StringReader MyStringReader) { /// <summary>正在分析的节</summary>
            ININode 当前节 = null; /// <summary>正在分析的项</summary>
            INIItem 当前项 = null; /// <summary>正在分析的节名</summary>
            string 当前节名 = null; /// <summary>正在分析的项名</summary>
            string 当前项名 = null; /// <summary>累计读取的属性行的计数</summary>
            int 计数 = 0; /// <summary>该行是合法有效的行,仍是没法识别的行。(没法识别做为备注处理)</summary>
            bool 有效行 = false; /// <summary>该行去掉空格和Tab符的文本长度</summary>
            int 有效文本长度; /// <summary>每一个实体前的注释</summary>
            string 备注 = ""; // * 循环读取每行内容 *
            while (true) { string 行文本 = MyStringReader.ReadLine(); if (行文本 == null) {  if (备注.Length > 0) { EndText = 备注; } 上次读取的有效行数 = 计数; break; } else { string 行; 有效行 = false; // * 获取 去掉空格和Tab符的文本 *
                    行 = 行文本.Trim(' ', '\t'); // * 获取 去掉空格和Tab符的文本的长度 *
                    有效文本长度 = 行.Length; // * 检测注释符 *
                    if (行文本.Contains(U注释)) { int 注释位置 = 行文本.IndexOf(U注释); 行 = 行文本.Substring(0, 注释位置); int 注释开始位置 = 注释位置 + U注释.Length - 1; int 注释长度 = 行文本.Length - 注释开始位置; if (注释长度 > 0) { if (备注.Length > 0) { 备注 += "\r\n"; } 备注 += 行文本.Substring(注释开始位置, 注释长度); } 有效行 = true; } if (行文本.Contains(U注释2)) { int 注释位置 = 行文本.IndexOf(U注释2); 行 = 行文本.Substring(0, 注释位置); int 注释开始位置 = 注释位置 + U注释2.Length - 1; int 注释长度 = 行文本.Length - 注释开始位置; if (注释长度 > 0) { if (备注.Length > 0) { 备注 += "\r\n"; } 备注 += 行文本.Substring(注释开始位置, 注释长度); } 有效行 = true; } // * 检查开头字符 *
                    if (行.Length >= 2) { //[类型定义]====首字符:U节首[
                        if (行[0] == U左括号[0]) { int 右括号位置 = 行.IndexOf(U右括号[0], 2); if (右括号位置 > 1) { 当前节名 = 行.Substring(1, 右括号位置 - 1); 当前节 = New(当前节名, 备注); 备注 = ""; 计数 += 1; 有效行 = true; } } //项定义====含有等号的行 // -> 获取赋值符号位置
                        int 赋值符位置 = 行.IndexOf(U等号, 2); if (赋值符位置 > 1) { // -> 得到名称和值,并新建项
                            当前项名 = 行.Substring(0, 赋值符位置).Trim(' ', '\t'); string 值 = 行.Substring(赋值符位置 + 1, 行.Length - 赋值符位置 - 1).Trim(' ', '\t'); if (当前节 != null) { 当前项 = 当前节.New(当前项名, 值, 备注); 备注 = ""; 计数 += 1; 有效行 = true; } } } // * 无效行做为备注处理 *
                    if (有效行 == false) { if (忽略备注 == false) { if (行文本.Length == 0) { 备注 += "\r\n"; } else { 备注 += 行文本 + "\r\n"; } } } } } return true; } #endregion

◇  编码问题

 

 1 /// <summary>
 2         /// 经过文件的头部开始的两个字节来区分一个文件属于哪一种编码。  3         /// 若是文件长度不足2字节,则返回null  4         /// 当FF FE时,是Unicode;  5         /// 当FE FF时,是BigEndianUnicode;  6         /// 当EF BB时,是UTF-8;  7         /// 当它不为这些时,则是ANSI编码。  8         /// </summary>
 9         public static Encoding GetFileEncodeType(string filename) { 10             FileStream fs = new FileStream(filename, FileMode.Open, FileAccess.Read); 11             BinaryReader br = new BinaryReader(fs); 12             Byte[] buffer = br.ReadBytes(2); 13             if (buffer.Length < 2) { return null; } 14             if (buffer[0] >= 0xEF) { 15                 if (buffer[0] == 0xEF && buffer[1] == 0xBB) { 16                     return Encoding.UTF8; 17                 } else if (buffer[0] == 0xFE && buffer[1] == 0xFF) { 18                     return Encoding.BigEndianUnicode; 19                 } else if (buffer[0] == 0xFF && buffer[1] == 0xFE) { 20                     return Encoding.Unicode; 21                 } else { 22                     return Encoding.Default; 23  } 24             } else { 25                 return Encoding.Default; 26  } 27         }

窗体读取INI演示

 ◇  演示效果

 

◇ INIListView类

用一个辅助类将INI文件内容显示到ListView来展示效果。

给每一个节点添加一个Group组,将节点自己和下辖的项都放进组。

当鼠标选中某项时,判断该item的Key和Group便可知道它属于哪一个节点,名称是什么。

 1     public class INIListView {  2         public ListView 视图;  3         public Color 节颜色 = Color.FromArgb(0, 153, 153);  4         public Color 节底色 = Color.FromArgb(255, 255, 255);  5         public void 绑定控件(ListView ListView) {  6             视图 = ListView;  7  初始化();  8  }  9         public void 载入数据(INI ini) { 10  初始化组(ini); 11  初始化数据(ini); 12  } 13 
14         private void 初始化() { 15             视图.View = View.Tile; 16             视图.ShowGroups = true; 17  初始化列(); 18  } 19 
20         private void 初始化列() { 21  视图.Columns.Clear(); 22             视图.Columns.Add("A", "名称", 220); 23             视图.Columns.Add("B", "", 300); 24             视图.Columns.Add("C", "注释", 440); 25  } 26         private void 初始化组(INI ini) { 27             if (ini == null) { return; } 28             for (int i = 0; i < ini.Nodes.Count; i++) { 29                 string nodeName = ini.Nodes[i].Name; 30                 int cc = ini.Nodes[i].Items.Count; 31                 string nodeTitle = string.Format("{0} ({1})", nodeName, cc); 32  视图.Groups.Add(nodeName, nodeTitle); 33  } 34  } 35 
36         private void 初始化数据(INI ini) { 37  视图.Items.Clear(); 38 
39             if (ini == null) { return; } 40             for (int i = 0; i < ini.Nodes.Count; i++) { 41                 string nodeName = ini.Nodes[i].Name; 42                 var nodeitem = 视图.Items.Add(nodeName, "["+nodeName+"]",0); 43                 nodeitem.ForeColor = 节颜色; 44                 nodeitem.BackColor = 节底色; 45          
46                 nodeitem.Group = 视图.Groups[nodeName]; 47                
48 
49                 for (int j = 0; j < ini.Nodes[i].Items.Count; j++) { 50                     var iniitem = ini.Nodes[i].Items[j]; 51                     string name = iniitem.Name; 52                     string value = iniitem.Value; 53                     string comment = iniitem.Comment; 54                     var item = 视图.Items.Add(name, name); 55                     item.Group = 视图.Groups[nodeName]; 56  item.SubItems.Add(value); 57  item.SubItems.Add(comment); 58  } 59  } 60  } 61 
62     }

窗体上拖一个ListView(数据视图)和OpenFileDialog(openINIFileDialog)、和Button(按钮_读取文件)

 1     public partial class 编辑窗体 : Form {  2         INIListView INIListView = new INIListView();  3  INI 当前文档;  4         
 5 
 6         public 编辑窗体() {  7  InitializeComponent();  8  }  9 
10         private void 编辑窗体_Load(object sender, EventArgs e) { 11             Width = 1280; 12             Height = 720; 13  初始化数据视图(); 14             openINIFileDialog.InitialDirectory = Environment.CurrentDirectory; 15  } 16         private void 初始化数据视图() { 17  INIListView.绑定控件(数据视图); 18  } 19 
20         private void 按钮_读取文件_Click(object sender, EventArgs e) { 21             var result = openINIFileDialog.ShowDialog(); 22             if (result == DialogResult.OK) { 23                 当前文档 = new INI(); 24                 var 读取结果 = 当前文档.读取文档(openINIFileDialog.FileName); 25  INIListView.载入数据(当前文档); 26  } 27 
28 
29  } 30 
31         private void 视图_1_Click(object sender, EventArgs e) { 32             数据视图.View = View.Details; 33  } 34 
35         private void 视图_2_Click(object sender, EventArgs e) { 36             数据视图.View = View.Tile; 37  } 38 
39         private void 视图_3_Click(object sender, EventArgs e) { 40             数据视图.View = View.List; 41  } 42 
43         private void 视图_4_Click(object sender, EventArgs e) { 44             数据视图.View = View.SmallIcon; 45  } 46 
47         private void 视图_5_Click(object sender, EventArgs e) { 48             数据视图.View = View.LargeIcon; 49  } 50     }

 

=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-

结语:本文实现了INI文件的构造、读取和写入。

实际上经过扩展能够实现更强大的数据格式。

相关文章
相关标签/搜索