首页 > 代码库 > 基于正则的INI读写工具类,支持加密解密

基于正则的INI读写工具类,支持加密解密

看到这个标题,有人会问,现在都用xml做配置文件了,谁还用INI文件啊!下面来简单对比一下xml和ini:

  1、XML功能强大表达能力强,同时扩展性好。
  2、它的主要优势是异构平台的整合、通讯。
  3、缺点主要是使用复杂,运行库占用的资源较多。
  4、如果多个程序进行数据交换或是跨平台通讯则使用功能强大的XML;

  5、INI虽表达能力不强,但是简单实用,接口方便。如果是用于应用程序的配置INI文件就够了。

至于哪个更好,应该用哪个,可以根据自己爱好和需求。个人感觉INI文件操作简单,就是读取文件,处理字符串,保存到文件,可谓是简单粗暴。而且内容也比较友好,没有冗余的东西。

  由于最近项目中用到INI文件,所以抽空编写了一个Helper,取名交INIHelper。这里先不给出它的源码,先来看下他的用法。

一、INIHelper的用法                                                                     

  这里为了做演示,我建了一个C# 控制台应用程序,随便起了个名字,加入了INIHelper这个类。项目结构如图:

在Debug目录下面添加了一个config.ini的文件,内容如下:

下面我们用这个Helper来读取这个INI文件的所有内容,代码如下:

class Program   {      static void Main(string[] args)      {         try         {            INIHelper helper = new INIHelper("config.ini");            Console.WriteLine(helper.GetValueByName("DBName"));            Console.WriteLine(helper.GetValueByName("UserName"));            Console.WriteLine(helper.GetValueByName("PassWord"));            Console.WriteLine(helper.GetValueByName("Version"));         }         catch (Exception ex)         {            Console.WriteLine(ex.Message);         }         Console.Read();      }   }

输出结果如下:

是不是很方便,这里还有另外一种写法,代码如下:

class Program   {      static void Main(string[] args)      {         try         {            INIHelper helper = new INIHelper();            helper.LoadINI("config.ini");            Console.WriteLine(helper.GetValueByName("DBName"));            Console.WriteLine(helper.GetValueByName("UserName"));            Console.WriteLine(helper.GetValueByName("PassWord"));            Console.WriteLine(helper.GetValueByName("Version"));         }         catch (Exception ex)         {            Console.WriteLine(ex.Message);         }         Console.Read();      }   }

代码中加粗的部分就是另外一种写法,一种方法是在构造时加载ini文件,另外一种方法时在需要的时候加载。到这里读取ini文件的就说完了,下面来说一下修改ini文件。这里我们来修改ini文件密码为root,然后保存到ini文件中,来看看代码怎么写:

class Program   {      static void Main(string[] args)      {         try         {            INIHelper helper = new INIHelper();            helper.LoadINI("config.ini");            helper.SetValueByName("PassWord", "root");            helper.SaveINI();         }         catch (Exception ex)         {            Console.WriteLine(ex.Message);         }         Console.Read();      }   }

首先加载ini文件,然后调用SetValueByName方法修改密码,最后调用SaveINI方法保存。保存后,可以打开ini文件看到内容变了,这里就不再截图了。其还支持加密解密,这样我们的配置文件内容就不会被被人看到和随意修改了,加密后的效果如下:

 

 

二、揭开INIHelper神秘的面纱                                                          

  下面来看看INIHelper的具体实现,首先来看构造方法和LoadINI,其实现代码如下:

      private string newLine = "\r\n";  //换行符      private string filePath = string.Empty; //文件名称      private string fileContent = string.Empty; //文件内容      public INIHelper() { }      /// <summary>      /// 有参构造方法,直接读取INI文件      /// </summary>      /// <param name="filePath"></param>      public INIHelper(string filePath)      {         this.LoadINI(filePath);      }      /// <summary>      /// 加载并读取INI文件      /// </summary>      /// <param name="fileName">文件路径</param>      public void LoadINI(string filePath)      {         if (filePath.Trim().Length > 0)         {            this.filePath = filePath;            ReadINIFile();         }         else         {            throw new Exception("Invalid file name!");         }      }

可以看到在有参构造方法里面调用了LoadINI方法,所以等价于调用无参构造函数然后调用LoadINI方法。LoadINI方法里面首先判断文件路径是否合法,合法的话就读取ini文件,否则抛出异常。ReadINIFile方法就是读取文件内容,然后赋给fileContent,其实现如下:

/// <summary>      /// 读取INI文件      /// </summary>      private void ReadINIFile()      {         if (File.Exists(this.filePath))         {            try            {               using (StreamReader sr = new StreamReader(this.filePath))               {                  this.fileContent = sr.ReadToEnd();                  this.fileContent = EncryptionAndDecryption(fileContent); //解密                  //如果文件内容为空或者没有换行符,则认为是无效的INI文件。                  if (fileContent.Trim().Length <= 0 || !fileContent.Contains("\n"))                  {                     throw new Exception("Invalid ini file");                  }                  else                  {                     //保存文件默认换行符                     if (!fileContent.Contains(newLine))                     {                        this.newLine = "\n";                     }                  }               }            }            catch (Exception ex)            {               throw new Exception("Read file error! Error Message:" + ex.Message);            }         }         else         {            throw new Exception("File " + filePath + " not found!");         }      }

这个已经包含了加密解密的方法,首先读取文件内容,解密,然后判断文件是否合法,及是否有为空和是否有换行符,然后判断里面的换行符是否为默认值,否则修改newLine为文件默认的换行符。(大家可以修改代码,自定分割符。默认是支持\r\n或\n)

/// <summary>      /// 读取INI文件某个配置项的值      /// </summary>      /// <param name="fieldName"></param>      /// <returns></returns>      public string GetValueByName(string fieldName)      {         fileContent = fileContent.Replace(newLine, ";");         fileContent = fileContent.Replace(" ", "");         fileContent = fileContent.EndsWith(";") ? fileContent : fileContent + ";";         Regex reg = new Regex("(?<=" + fieldName + "=).*?(?=;)");         Match m = reg.Match(fileContent);         return m.Value;      }      /// <summary>      /// 修改INI文件某个配置项的值      /// </summary>      /// <param name="fieldName"></param>      /// <param name="value"></param>      public void SetValueByName(string fieldName, string value)      {         string reg = "(?<=" + fieldName + "=).*?(?=;)";         fileContent = Regex.Replace(fileContent, reg, value);      }

这个是读取和修改某个配置项的方法,使用正则表达式进行匹配。修改只是修改fileContent的值,并不执行保存。

/// <summary>      /// 保存对INI文件的修改      /// </summary>      public void SaveINI()      {         try         {            fileContent = fileContent.Replace(";", newLine); //替换换行符            fileContent = EncryptionAndDecryption(fileContent); //加密            using (StreamWriter sw = new StreamWriter(filePath))            {               sw.Write(fileContent);               sw.Close();            }         }         catch (Exception ex)         {            throw new Exception("Save file error! Error Message:" + ex.Message);         }      }      /// <summary>      /// 加密解密算法,使用异或算法      /// </summary>      /// <param name="str"></param>      public string EncryptionAndDecryption(string str)      {         byte key = 32;         byte[] buffer = Encoding.Default.GetBytes(str);         for (int i = 0; i < buffer.Length; i++)         {            buffer[i] ^= key;         }         return Encoding.Default.GetString(buffer);      }

SaveINI执行加密后保存到ini文件,这里给出了简单的对称加密算法,大家使用时可以使用自定义的加密算法。

注意:笫一次读取配置文件由于没有加密,调用了解密算法,所以会出现文件无效的异常。这里需要先加密保存一次,然后就好了。

 

源码下载

 

基于正则的INI读写工具类,支持加密解密