原文:http://www.manew.com/thread-105598-1-1.html
引言
1、您是否需要在项目中使用Txt、Excel等表格配置?
2、您是否还在一行行写配置解析代码?
3、您是否担心运行时解析速度?
本文将一一解答上面的问题,帮助你更好地使用配置
源码:https://github.com/RickJiangShu/ConfigManager
范例工程:https://github.com/RickJiangShu/ConfigManager-Example
设计思路
1、自动生成解析类
项目中经常需要用到策划配置,而一个策划配置需要写一个解析类去解析,这样浪费了大量的时间且容易出错。
因此,我觉得可以写一个自动生成解析类的工具。
2、编辑器下解析
通常解析工作是放在运行时,即加载一个文本文件之后再进行解析这个文本。这样在数据量巨大的时候,解析速度堪忧。
因此,我想把大量的解析工作放在编辑器下去处理。
流程图
整个流程如上图所示,所以整个思路可以分为以下四个部分:解析、反射、序列化和反序列化
解析
1. 按“分隔符”与“换行符”将表格切割成“行x列”的字符串矩阵,关键代码如下:
[C#]
纯文本查看
复制代码
public static string[,] Content2Matrix(string config, string sv, string lf, out int row, out int col)
{
config = config.Trim();//清空末尾的空白
//分割
string[] lines = Regex.Split(config, lf);
string[] firstLine = Regex.Split(lines[0], sv, RegexOptions.Compiled);
row = lines.Length;
col = firstLine.Length;
string[,] matrix = new string[row, col];
//为第一行赋值
for (int i = 0, l = firstLine.Length; i < l; i++)
{
matrix[0, i] = firstLine;
}
//为其他行赋值
for (int i = 1, l = lines.Length; i < l; i++)
{
string[] line = Regex.Split(lines, sv);
for (int j = 0, k = line.Length; j < k; j++)
{
matrix[i, j] = line[j];
}
}
return matrix;
}
2.从矩阵中取出相应的字符,替换自定义模板中的变量并写入文件,关键代码如下:
[C#]
纯文本查看
复制代码
string idType = ConfigTools.SourceType2CSharpType(src.matrix[1, 0]);
string idField = src.matrix[2, 0];
//属性声明
string declareProperties = "";
for (int x = 0; x < src.column; x++)
{
string comment = src.matrix[0, x];
string csType = ConfigTools.SourceType2CSharpType(src.matrix[1, x]);
string field = src.matrix[2, x];
string declare = string.Format(templete2, comment, csType, field);
declareProperties += declare;
}
//替换
content = content.Replace("/*ClassName*/", src.configName);
content = content.Replace("/*DeclareProperties*/", declareProperties);
content = content.Replace("/*IDType*/", idType);
content = content.Replace("/*IDField*/", idField);
//写入
ConfigTools.WriteFile(outputPath, content);
生成的C#文件如下
反射
上面解析出了C#文件和一个SerializableSet.cs,接下来将通过反射特性实例化一个SerializableSet对象,关键代码如下:
[C#]
纯文本查看
复制代码
public static object Serialize(List<Source> sources)
{
Type t = FindType("SerializableSet");
if (t == null)
{
UnityEngine.Debug.LogError("找不到SerializableSet类!");
return null;
}
object set = UnityEngine.ScriptableObject.CreateInstance(t);
foreach(Source source in sources)
{
string fieldName = source.sourceName + "s";
Array configs = Source2Configs(source);
FieldInfo fieldInfo = t.GetField(fieldName);
fieldInfo.SetValue(set,configs);
}
return set;
}
序列化
最后就是使用Unity API创建Asset文件,关键代码如下:
[C#]
纯文本查看
复制代码
UnityEngine.Object set = (UnityEngine.Object)Serializer.Serialize(sources);
string o = cache.assetOutputFolder + "/" + assetName;
AssetDatabase.CreateAsset(set, o);
生成的文件如下:
反序列化
因为要在运行时使用,所以反序列化的代码没有使用反射(效率低)。而是在解析的过程中解析出一个反序列化文件,生成的代码如下:
[C#]
纯文本查看
复制代码
public class Deserializer
{
public static void Deserialize(SerializableSet set)
{
for (int i = 0, l = set.Equips.Length; i < l; i++)
{
EquipConfig.GetDictionary().Add(set.Equips.EquipId, set.Equips);
}
for (int i = 0, l = set.EquipCSVs.Length; i < l; i++)
{
EquipCSVConfig.GetDictionary().Add(set.EquipCSVs.EquipId, set.EquipCSVs);
}
}
}
最后,直接调用相应的Config.Get(id)即可,效果如下:
若您发现其中有问题或任何优化意见,请在本帖留言或私信我,谢谢大家~