zoukankan      html  css  js  c++  java
  • C# 如何实现完整的INI文件读写类

    作者: 魔法软糖

    日期: 2020-02-27

    引言

    *************************************

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

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

    在简单只需要读取的场合,调用WINDOWS API就行,但在复杂的需要可视化编辑时,就需要建立自己的处理类了。

     

    如何实现自己的INI类

    首先我们需要了解

    ◇  INI的格式

     

    ◇  典型INI文件

    ;项目注释
    [.ShellClassInfo]
    InfoTip=有图标的文件夹
    ;图标资源
    IconResource="C:Windowssystem32SHELL32.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("
    ", "  ")); }
    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(' ', '	');
                        // * 获取 去掉空格和Tab符的文本的长度 *
                        有效文本长度 = 行.Length;
                        // * 检测注释符 *
                        if (行文本.Contains(U注释)) {
                            int 注释位置 = 行文本.IndexOf(U注释);
                            行 = 行文本.Substring(0, 注释位置);
                            int 注释开始位置 = 注释位置 + U注释.Length - 1;
                            int 注释长度 = 行文本.Length - 注释开始位置;
                            if (注释长度 > 0) {
                                if (备注.Length > 0) { 备注 += "
    "; }
                                备注 += 行文本.Substring(注释开始位置, 注释长度);
                            }
                            有效行 = true;
                        }
                        if (行文本.Contains(U注释2)) {
                            int 注释位置 = 行文本.IndexOf(U注释2);
                            行 = 行文本.Substring(0, 注释位置);
                            int 注释开始位置 = 注释位置 + U注释2.Length - 1;
                            int 注释长度 = 行文本.Length - 注释开始位置;
                            if (注释长度 > 0) {
                                if (备注.Length > 0) { 备注 += "
    "; }
                                备注 += 行文本.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(' ', '	');
                                string 值 = 行.Substring(赋值符位置 + 1, 行.Length - 赋值符位置 - 1).Trim(' ', '	');
                                if (当前节 != null) {
                                    当前项 = 当前节.New(当前项名, 值, 备注);
                                    备注 = "";
                                    计数 += 1;
                                    有效行 = true;
                                }                                                      
                            }
                        }
                        // * 无效行作为备注处理 *
                        if (有效行 == false) {
                            if (忽略备注 == false) {
                                if (行文本.Length == 0) { 备注 += "
    "; } else { 备注 += 行文本 + "
    "; }
                            }
                        }
                    }                             
                }
                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文件的构造、读取和写入。

    实际上通过扩展可以实现更强大的数据格式。

  • 相关阅读:
    CPU的物理限制
    递归快还是循环(迭代)快?
    VS2010下测试程序性能瓶颈
    Qt编程之实现在QFileDialog上添加自定义的widget
    This application failed to start because it could not find or load the Qt platform plugin "windows"
    网络设备Web登录检测工具device-phamer
    Outlook数据提取工具readpst
    Xamarin无法调试Android项目
    Web应用扫描工具Wapiti
    Xamarin 2017.11.1更新
  • 原文地址:https://www.cnblogs.com/createwell/p/12715391.html
Copyright © 2011-2022 走看看