Unity导表工具Luban插件的数据加载原理与优化
创始人
2024-01-25 09:20:49
0

文章目录

    • 1. 配置Luban并测试
    • 2. Luban 数据加载原理
    • 3. 根据工程进行懒加载优化
      • 3.1 增加公共接口与接口实现方法
      • 3.2 外部提供获取数据的方法
      • 3.3 对Luban中的导表模板进行修改
    • 4. 优化后测试


如果初入坑,Luban配置可以参考宝鱼大佬的视频教程 强力配置工具luban使用教程【01】

或者查看官方文档 快速开始Luban

1. 配置Luban并测试

我这里是已经配置好的unity工程,数据使用的是官方案例里的 MiniTemplate。为了方便演示,除了生成一份C#代码之外,额外生成一份Json数据。

--gen_types code_cs_unity_json,data_json  ^ 

代码生成后,自定义增加一个Test脚本进行测试

using System.IO;
using cfg;
using UnityEngine;
using SimpleJSON;public class Test : MonoBehaviour
{void Start(){Tables tab = new Tables(Loader);foreach (var n in tab.TbItem.DataList){print($"ID:{n.Id};名称:{n.Name};描述:{n.Desc}");}}private static JSONNode Loader(string file){string path = "Assets/Res/Datas/" + file + ".json";return JSON.Parse(File.ReadAllText(path, System.Text.Encoding.UTF8));}
}

控制台看到如下输出,说明Luban已经配置ok。

在上面的测试代码里,可以看到当我需要使用Excel数据时,得先去new一个Tables对象,然后通过对象小数点后ClassName.DataList 或者 ClassName.DataMap 拿到我想要的数据。除此之外,new Tables() 在构造时还需要传入一个装载器,这个装载器去获取数据源与tables里的各个类对象进行匹配

2. Luban 数据加载原理

需要了解Luban加载数据的工作原理,我们就需要先来看一下 Tables 的构造方法(Tables.cs时自动生成的脚本,在导出目录根路径下)

下图是默认模板导出的Tables类,如果你没有使用自定义模板,应该和我一样。
在这里插入图片描述

下图是TbItem表 (表示整张表)内的构造函数
在这里插入图片描述

下图是Item类(表示每一行数据结构)生成后的构造函数

圈重点,通过上面的三个构造方法,可以得知导表后:

Tables会将已经生成好的表对象(本案例指TbItem)作为自己的一个参数,在构造时通过Loader回调加载源数据的方式(生成的Json数据或者二进制数据)去遍历表中每一行装载数据

现在只导入了一张表,如果多张表Tables会做什么呢?不妨多加几张表来看看。

如上图,我在
__tables__.xlsx
中新增了两张测试表并导出,再来看一下 Tables 的构造方法。

Tables中的对象增加了,也就是说在new Tables时,它会全量加载,把所有的表对象都new一遍,获取源数据一对一装载,并把他们以表名为key的方式添加进字典里。

目前来看,Luban的数据加载方式需要我们在数据管理类中持有Tables对象,并且在具体使用到某张表数据之前进行Tables的预加载。

3. 根据工程进行懒加载优化

由于luban功能强大,所以全量可以支持它这么做。但是对于一般的工程来说,尤其是在游戏开发中拥有海量的数据源,我在初始化时全量加载反而弊大于利,并且为了方便数据管理和程序调用,也用不到那么多的功能。基于此,这里我们可以把这种预加载方式修改为懒加载。

3.1 增加公共接口与接口实现方法

定义一个VO接口,内部提供一个数据加载的方法,我这里传入json.text,可自定义。

public interface IVOFun
{void _LoadData(string data);
}

对每一个生成的表对象,让他们继承IVOFun接口,并在其内部实现_LoadData方法

这里以默认unity_json案例中的TbItem为例,在它内部实现LoadData

public void _LoadData(string data)
{JSONNode _json = JSON.Parse(data);foreach(JSONNode _row in _json.Children){var _v = item.Item.DeserializeItem(_row);_dataList.Add(_v);_dataMap.Add(_v.Id, _v);}PostInit();
}

3.2 外部提供获取数据的方法

在工程中你需要获取数据的地方添加新增Get数据的方法,我这里使用的资源加载方式时Addressables,代码实现原理都一样,根据自己情况只是获取数据源Data的方式不同,但需要注意的是这里应该使用异步加载与拿到data后释放资源

这里的思路是定义一个字典 Dictionary<源数据名称,数据对象>,需要加载数据对象时先从字典中获取,如果没有再new并加载数据后存入字典中。

private const string VO_DATA_PATH = "Res/Datas/";
readonly Dictionary tables = new Dictionary();private TextAsset ReadData(string path)
{AsyncOperationHandle op = Addressables.LoadAssetAsync(path);TextAsset res = op.WaitForCompletion();Addressables.Release(op);return res;
}public T GetVOData(string fileName) where T : IVOFun, new()
{var path = VO_DATA_PATH + fileName + ".json";if (tables.ContainsKey(fileName)){return (T)tables[fileName];}else{var data = new T();TextAsset text = ReadData(path);data._LoadData(text.text);tables.Add(fileName, data);return data;}
}

这样一来,便可以在工程中任何节点加载需要使用到的资源。DataManager.Instance 是你提供获取数据方法的地方,我这里是一个单例类。

TbItem vo = DataManager.Instance.GetVOData("itemdata");
var infos = vo.DataList;
foreach (var n in infos) print($"{n.Name}/{n.Desc}");

当然 GetVOData 也可以完全不需要传入数据源的文件名称,这一步骤可以将接口的 _LoadData 方法修改为不需要传参,让各个表内部自己去获取与自身绑定的数据源。

传参可以额外让两份相同的结构的数据使用同一个表对象,不传参也可以更加便捷,让程序更多的关注在数据使用方面,而不是还需要额外知道数据源的文件名称。这里只是提供一种思路,可以根据自身的使用情况去自定义内部实现。

基于此,已经完成了懒加载步骤的70%,剩下的步骤需找到与模板相对应的文件目录对表定义进行修改。我这里的生成模板是code_cs_unity_json,所以来到Luban.ClientServer/Templates/config/cs_unity_json目录(该目录下存放了unity的C#脚本生成模板)。如果你是C#二进制模板,应该前往cs_bin目录。

3.3 对Luban中的导表模板进行修改

截图时,我把行号也截图进去了,修改的位置大致在截图中行号附近。其他类型修改位置还需要自己去详细看一下模板定义。

第一步:修改 tables.tpl

在命名空间与表名之间增加上面定义的接口 IVOFun

在这里插入图片描述
除此之外,还可以去掉冗余代码,将Tables的构造方法置为空。

在这里插入图片描述

替换后代码如下图所示:

在这里插入图片描述

第二步:修改 table.tpl

在类名后增加继承接口 IVOFun

在这里插入图片描述

修改table构造函数
在这里插入图片描述

在末尾增加接口实现方法
在这里插入图片描述

至此,已经完全完成了懒加载的修改,可以重新导出脚本了。

4. 优化后测试

完成第3节后,需要重新导入一次,我这里新建一个测试脚本。

using cfg.item;
using UnityEngine;public class LazyTest : MonoBehaviour
{void Start(){TbItem vo = DataManager.Instance.GetVOData("itemdata");var infos = vo.DataList;foreach (var n in infos){print($"{n.Name}/{n.Desc}");}}
}

运行游戏,可以看到控制台成功打印信息。

在这里插入图片描述


Author:文若
DataTime:2022-11-16

相关内容

热门资讯

“世界市长对话·青岛”聚焦文旅...   新华社青岛7月19日电(记者 丛佳鑫、李傲秋)7月19日,以“精彩文旅 友城共享”为主题的“世界...
AI赋能千行百业,创新潮涌八桂...   来源:7月20日《新华每日电讯》  作者:新华每日电讯记者王军伟 黄浩铭 黄耀滕 王楚然  “人...
​现金流量表怎么编制 现金流量表怎么编制编制现金流量表的方法主要包括直接法和间接法两种。直接法通过列出主要类别的现金收入和...
【理响中国·一线探新】解锁楼宇...   编者按:楼宇经济在城市经济发展中具有举足轻重的作用,它通过垂直空间集约利用,形成高密度、高附加值...
“AI超级联赛”来了,为何选择...   新华社南宁7月19日电(记者陈露缘、赵欢)奖金丰厚、赛道多样,还可跨国组队,在全国各地AI浪潮兴...
链博会外国参展商:跟中国做生意...   第三届中国国际供应链促进博览会上,有不少外国参展商前来参会。链博会会期已过半,这些外国参展商都有...
“世界市长对话·青岛”:历史城...   中新网青岛7月19日电(胡耀杰 张孝鹏)“世界市长对话·青岛”19日在山东青岛举行,吸引来自白俄...
我的青春我“职”道 | “苹果...   编者按:  2025年6月30日-7月4日,由中国社会科学院大学和云南民族大学主办,云南民族大学...
98+1!新央企中国雅江集团正...   中国经济网北京7月19日讯 国资委今日发布公告,经国务院批准,组建中国雅江集团有限公司,由国务院...
曝光10年仍在售!记者卧底揭秘...   编者按:这里是民生调查局,见人所未见,调查民生之变。关注你想关注的、你没关注的,调查你想看的、未...