[Unity3D·CSV篇]02.CSV高手级读取

笨木头  2017-06-11 11:20   Unity3D   阅读(58)   0条评论
转载请注明,原文地址:http://www.benmutou.com/archives/2346
文章来源:笨木头与游戏开发
 
在上一篇中,保存CSV文件每一行的数据为CSVDemo对象时,总是要用if条件判断当前是哪个字段,然后再给对象赋值。
这样太烦人了,要是在开发的过程中不断地有新的想法,不断地新增修改字段,那真是改死人了。
甚至完全失去了改动的兴趣,于是游戏的创意就大打折扣,后果十分严重,嗯嗯。(旁白:说了这么多,接下来就要宣传你的“产品”了吧?
 
不过,大家不用怕,有木头在,木头有个好招,大家看招吧。
 

1. 临时的字典结构

为了以后更方便,我们在搭建代码的时候,通常都会麻烦一点,但是,一劳永逸的事情,我是不会抗拒的。

为了实现我们的目的,我们需要先用一个字典结构把CSV文件保存起来,如下代码:


  1.     /// <summary>
  2.     /// 读取CSV文件
  3.     /// 结果保存到字典集合,以ID作为Key值,对应每一行的数据,每一行的数据也用字典集合保存。
  4.     /// </summary>
  5.     /// <param name="filePath"></param>
  6.     /// <returns></returns>
  7.     public static Dictionary<string, Dictionary<string, string>> LoadCsvFile(string filePath)
  8.     {
  9.         Dictionary<string, Dictionary<string, string>> result = new Dictionary<string, Dictionary<string, string>>();
  10.         string[] fileData = File.ReadAllLines(filePath);
  11.         /* CSV文件的第一行为Key字段,第二行开始是数据。第一个字段一定是ID。 */
  12.         string[] keys = fileData[0].Split(',');
  13.         for (int i = 1; i < fileData.Length; i++)
  14.         {
  15.             string[] line = fileData[i].Split(',');
  16.             /* 以ID为key值,创建一个新的集合,用于保存当前行的数据 */
  17.             string ID = line[0];
  18.             result[ID] = new Dictionary<string, string>();
  19.             for (int j = 0; j < line.Length; j++)
  20.             {
  21.                 /* 每一行的数据存储规则:Key字段-Value值 */
  22.                 result[ID][keys[j]] = line[j];
  23.             }
  24.         }
  25.         return result;
  26.     }
实际上大部分代码和上一篇是相同的,主要的改动解释如下:
a. 暂时抛弃了CSVDemo类,把CSV文件以【Dictionary<string, Dictionary<string, string>>】的形式保存起来
b. 相当于是,以每一行的ID作为key值,value值保存的是每一行的所有数据
c. 每一行的所有数据又由一个Dictionary<string, string>保存,key值是字段名,value值是每一列的数据值
d. 最终,每一行都以这样的形式保存起来
 
这样保存起来的结构并不是很方便,因为类型都是字符串,不好取值,没关系,这个只是过度的。
 

2. 强大的反射

你是否曾经有这样的感觉,那些看起来明明一样的代码,仅仅只是类型不一样;那些看起来明明是一样的代码,仅仅只是调用的属性不一样。
总之,那些看起来明明不应该那么重复的代码,它却总是重复了。

 

大多数开发者都会有这样的感觉,于是,逆天的反射和泛型出现在我们的面前。
没错,木头要用的就是反射+泛型这个强力的组合。

 

看看优化后的代码:

  1.         /* 把CSV文件按行存放,每一行的ID作为key值,内容作为value值 */
  2.         Dictionary<int, CSVDemo> csvDataDic = new Dictionary<int, CSVDemo>();
  3.         /* CSV文件路径 */
  4.         string filePath = Application.streamingAssetsPath + "/CSVDemo.csv";
  5.         /* 从CSV文件读取数据 */
  6.         Dictionary<string, Dictionary<string, string>> datasDic = LoadCsvFile(filePath);
  7.         /* 遍历每一行数据 */
  8.         foreach (string ID in datasDic.Keys)
  9.         {
  10.             /* CSV的一行数据 */
  11.             Dictionary<string, string> datas = datasDic[ID];
  12.             /* 读取Csv数据对象的属性 */
  13.             PropertyInfo[] props = typeof(CSVDemo).GetProperties();
  14.             /* 使用反射,将CSV文件的数据赋值给CSV数据对象的相应字段,要求CSV文件的字段名和CSV数据对象的字段名完全相同 */
  15.             CSVDemo obj = new CSVDemo();
  16.             foreach (PropertyInfo pi in props)
  17.             {
  18.                 pi.SetValue(obj, Convert.ChangeType(datas[pi.Name], pi.PropertyType), null);
  19.             }
  20.             /* 按ID-数据的形式存储 */
  21.             csvDataDic[obj.ID] = obj;
  22.         }
主要修改如下:
a. 利用之前写的LoadCsvFile函数获取到字典结构的CSV文件内容
b. 遍历字典结构,取出每一行的内容
c. 利用反射获取CSVDemo类的所有属性
d. 便利CSVDemo类的所有属性,通过每一个属性名字到字典里获取数据内容,然后给CSVDemo对象相应的属性赋值
e. pi.SetValue就是用来给CSVDemo对象的某个属性赋值的,Convert.ChangeType是为了把字符串内容转换为属性对应的类型

 

最后来测试一下:

  1.         /* 测试读取ID为1的数据 */
  2.         CSVDemo csvDemo1 = csvDataDic[1];
  3.         Debug.Log("ID=" + csvDemo1.ID + ",Name=" + csvDemo1.Name);
尝试获取ID为1的那一行数据,输出日志如下:
结果是完全一样的,但是我们不需要再一个个if条件去判断然后赋值了,反射帮我们做了这件事情。(旁白:那泛型呢?

3. 泛型呢?

大家一定还有疑问,每个CSV文件都要写这样一段代码来读取吗?
不,这种重复的事情木头是绝对不会做的,我一做重复的事情就会头晕,于是,泛型救了我。
最终完美的读取并保存CSV文件的函数是这样的:

  1.     /// <summary>
  2.     /// 读取CSV文件数据(利用反射)
  3.     /// </summary>
  4.     /// <typeparam name="CsvData">CSV数据对象的类型</typeparam>
  5.     /// <param name="csvFilePath">CSV文件路径</param>
  6.     /// <param name="csvDatas">用于缓存数据的字典</param>
  7.     /// <returns>CSV文件所有行内容的数据对象</returns>
  8.     private Dictionary<int, T_CsvData> LoadCsvData<T_CsvData>(string csvFilePath)
  9.     {
  10.         Dictionary<int, T_CsvData> dic = new Dictionary<int, T_CsvData>();
  11.         /* 从CSV文件读取数据 */
  12.         Dictionary<string, Dictionary<string, string>> result = LoadCsvFile(csvFilePath);
  13.         /* 遍历每一行数据 */
  14.         foreach (string ID in result.Keys)
  15.         {
  16.             /* CSV的一行数据 */
  17.             Dictionary<string, string> datas = result[ID];
  18.             /* 读取Csv数据对象的属性 */
  19.             PropertyInfo[] props = typeof(T_CsvData).GetProperties();
  20.             /* 使用反射,将CSV文件的数据赋值给CSV数据对象的相应字段,要求CSV文件的字段名和CSV数据对象的字段名完全相同 */
  21.             T_CsvData obj = Activator.CreateInstance<T_CsvData>();
  22.             foreach (PropertyInfo pi in props)
  23.             {
  24.                  pi.SetValue(obj, Convert.ChangeType(datas[pi.Name], pi.PropertyType), null);
  25.             }
  26.             /* 按ID-数据的形式存储 */
  27.             dic[Convert.ToInt32(ID)] = obj;
  28.         }
  29.         return dic;
  30.     }
基本上把CSVDemo类型改为T_CsvData泛型即可,调用方式如下:

  1.         /* 把CSV文件按行存放,每一行的ID作为key值,内容作为value值 */
  2.         Dictionary<int, CSVDemo> csvDataDic2 = LoadCsvData<CSVDemo>(filePath);
  3.         /* 测试读取ID为2的数据 */
  4.         CSVDemo csvDemo2 = csvDataDic2[2];
  5.         Debug.Log("ID=" + csvDemo2.ID + ",Name=" + csvDemo2.Name);
输出的日志如下:

怎么样?以后读取CSV文件,只需调用【LoadCsvData<CSVDemo>(filePath);】即可。
 

4. 缓存?

大家肯定还有疑问,每次要用到CSV文件的数据时,都加载一遍,太没效率了吧?
你说得对,这样太没效率了,所以下一篇,木头要开始聊聊如何用最帅的方式缓存CSV文件对象了。
 
 

------点一点广告,支持笨木头------

发表评论

电子邮件地址不会被公开。 必填项已用*标注