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

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

1. 临时的字典结构

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

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


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
    /// <summary>
    /// 读取CSV文件
    /// 结果保存到字典集合,以ID作为Key值,对应每一行的数据,每一行的数据也用字典集合保存。
    /// </summary>
    /// <param name="filePath"></param>
    /// <returns></returns>
    public static Dictionary<string, Dictionary<string, string>> LoadCsvFile(string filePath)
    {
        Dictionary<string, Dictionary<string, string>> result = new Dictionary<string, Dictionary<string, string>>();

        string[] fileData = File.ReadAllLines(filePath);

        /* CSV文件的第一行为Key字段,第二行开始是数据。第一个字段一定是ID。 */
        string[] keys = fileData[0].Split(',');

        for (int i = 1; i < fileData.Length; i++)
        {
            string[] line = fileData[i].Split(',');
            /* 以ID为key值,创建一个新的集合,用于保存当前行的数据 */

            string ID = line[0];
            result[ID] = new Dictionary<string, string>();

            for (int j = 0; j < line.Length; j++)
            {
                /* 每一行的数据存储规则:Key字段-Value值 */
                result[ID][keys[j]] = line[j];
            }
        }
        return result;
    }
实际上大部分代码和上一篇是相同的,主要的改动解释如下:
a. 暂时抛弃了CSVDemo类,把CSV文件以【Dictionary<string, Dictionary<string, string>>】的形式保存起来
b. 相当于是,以每一行的ID作为key值,value值保存的是每一行的所有数据
c. 每一行的所有数据又由一个Dictionary<string, string>保存,key值是字段名,value值是每一列的数据值
d. 最终,每一行都以这样的形式保存起来
 
这样保存起来的结构并不是很方便,因为类型都是字符串,不好取值,没关系,这个只是过度的。
 

2. 强大的反射

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

 

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

 

看看优化后的代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
        /* 把CSV文件按行存放,每一行的ID作为key值,内容作为value值 */
        Dictionary<int, CSVDemo> csvDataDic = new Dictionary<int, CSVDemo>();

        /* CSV文件路径 */
        string filePath = Application.streamingAssetsPath + "/CSVDemo.csv";

        /* 从CSV文件读取数据 */
        Dictionary<string, Dictionary<string, string>> datasDic = LoadCsvFile(filePath);

        /* 遍历每一行数据 */
        foreach (string ID in datasDic.Keys)
        {
            /* CSV的一行数据 */
            Dictionary<string, string> datas = datasDic[ID];

            /* 读取Csv数据对象的属性 */
            PropertyInfo[] props = typeof(CSVDemo).GetProperties();

            /* 使用反射,将CSV文件的数据赋值给CSV数据对象的相应字段,要求CSV文件的字段名和CSV数据对象的字段名完全相同 */
            CSVDemo obj = new CSVDemo();
            foreach (PropertyInfo pi in props)
            {
                pi.SetValue(obj, Convert.ChangeType(datas[pi.Name], pi.PropertyType), null);
            }

            /* 按ID-数据的形式存储 */
            csvDataDic[obj.ID] = obj;
        }
主要修改如下:
a. 利用之前写的LoadCsvFile函数获取到字典结构的CSV文件内容
b. 遍历字典结构,取出每一行的内容
c. 利用反射获取CSVDemo类的所有属性
d. 便利CSVDemo类的所有属性,通过每一个属性名字到字典里获取数据内容,然后给CSVDemo对象相应的属性赋值
e. pi.SetValue就是用来给CSVDemo对象的某个属性赋值的,Convert.ChangeType是为了把字符串内容转换为属性对应的类型

 

最后来测试一下:

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

3. 泛型呢?

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
    /// <summary>
    /// 读取CSV文件数据(利用反射)
    /// </summary>
    /// <typeparam name="CsvData">CSV数据对象的类型</typeparam>
    /// <param name="csvFilePath">CSV文件路径</param>
    /// <param name="csvDatas">用于缓存数据的字典</param>
    /// <returns>CSV文件所有行内容的数据对象</returns>
    private Dictionary<int, T_CsvData> LoadCsvData<T_CsvData>(string csvFilePath)
    {
        Dictionary<int, T_CsvData> dic = new Dictionary<int, T_CsvData>();

        /* 从CSV文件读取数据 */
        Dictionary<string, Dictionary<string, string>> result = LoadCsvFile(csvFilePath);

        /* 遍历每一行数据 */
        foreach (string ID in result.Keys)
        {
            /* CSV的一行数据 */
            Dictionary<string, string> datas = result[ID];

            /* 读取Csv数据对象的属性 */
            PropertyInfo[] props = typeof(T_CsvData).GetProperties();

            /* 使用反射,将CSV文件的数据赋值给CSV数据对象的相应字段,要求CSV文件的字段名和CSV数据对象的字段名完全相同 */
            T_CsvData obj = Activator.CreateInstance<T_CsvData>();

            foreach (PropertyInfo pi in props)
            {
                 pi.SetValue(obj, Convert.ChangeType(datas[pi.Name], pi.PropertyType), null);
            }

            /* 按ID-数据的形式存储 */
            dic[Convert.ToInt32(ID)] = obj;
        }
        return dic;
    }
基本上把CSVDemo类型改为T_CsvData泛型即可,调用方式如下:

1
2
3
4
5
6
7
        /* 把CSV文件按行存放,每一行的ID作为key值,内容作为value值 */
        Dictionary<int, CSVDemo> csvDataDic2 = LoadCsvData<CSVDemo>(filePath);

        /* 测试读取ID为2的数据 */
        CSVDemo csvDemo2 = csvDataDic2[2];

        Debug.Log("ID=" + csvDemo2.ID + ",Name=" + csvDemo2.Name);
输出的日志如下:

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

4. 缓存?

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

发表评论

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