博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
014-热更新之xlua语言一
阅读量:6614 次
发布时间:2019-06-24

本文共 5354 字,大约阅读时间需要 17 分钟。

终于我们把lua语言的部分结束了,今天我们正式开始xlua的学习。首先我们先看看热更新是什么?热:就是刚出炉。简单来说就是当游戏某个功能出现bug,或者修改了某个功能的时候,或者增加了某个功能的时候,我们不需要重新下载安装安装包,就可以更新游戏内容。热更新的好处“不用浪费流量重新下载,不用通过商店审核更加快速,不用重新安装玩家可以更快体验到更新的内容。目前比较首欢迎的热更新方案:ulua tolua xlua。好了,因为我们在前面已经学习过lua语言了,所以我们的前言就是这些了,就不说太多的废话了。

我们首先创建一个unity工程,并将xlua的asset导入其中。说明一下,我们是根据作者提供的教程来学习的。

一、lua文件加载

1.执行字符串 最基本是直接用LuaEnv.DoStrig执行一个字符串。我们在空的物体上添加一个c#,并写如下的代码:

private LuaEnv luaenv;	void Start () {        luaenv = new LuaEnv();        //方式一        luaenv.DoString("print('Hello World!')");        //方式二        luaenv.DoString("CS.UnityEngine.Debug.Log('Hello World!')");        luaenv.Dispose();	}

 就会有如下图的效果:

 

其实这个算法也是没简单的,只要将lua虚拟机创建好就行了,但是作者他不推荐这个方式。

2.加载lua文件

用lua的require函数就行了。这个require我们还是比较熟悉的,因为我们在lua语言中的模块一节中,我们已经学过了,形式是这样的:require('文件名'),所以我们在xlua中也是可以这样写的。require实际上是调用一个个的loader去加载,有一个成功就不再往下尝试,全失败则包文件找不到。目前xlua除了原生的loader外,还添加了从Resource加载的loader,需要注意的是因为Resource只支持有限的后缀,放Resources下的lua文件的加上txt后缀。我们需要在Resources文件下创建一个txt文件,名字叫helloworld.lua.txt,在其中写print('HelloWorld'),就能完成与上面一样的操作了。代码如下:

void Start () {        TextAsset ta = Resources.Load
("helloworld.lua"); LuaEnv luaenv = new LuaEnv(); luaenv.DoString(ta.ToString()); luaenv.Dispose(); }

3.自定义loader

在xlua加自定义loader是很简单的,只涉及到一个接口:public delegate byte[] CustomLoader(ref string filepath);public void LuaEnv.AddLoader(CustomLoader loader)。通过AddLoader可以注册多个回调,该回调参数是字符串,lua代码里头调用require时,参数将会透传给回调,回调中就可以根据这个参数支持调试,需要把filepath修改为真实路径传出。该回调返回值是一个byte数组,如果为空表示该loader找不到,否则则为lua文件的内容。

我们首先在unity创建一个StreamingAssets文件,在其中创建一个txt文件,名叫text007.lua.txt。print('I am 007'),接着我们在空游戏物体上的c#脚本上添加如下的脚本。

using UnityEngine;using XLua;using System.IO;public class DefineLoader : MonoBehaviour {    // Use this for initialization    private LuaEnv luaenv;	void Start () {        luaenv = new LuaEnv();        luaenv.AddLoader(MyLoader);        luaenv.DoString(" require 'text007'");        luaenv.Dispose();	}    private byte[] MyLoader(ref string filPath)    {        string absPath = Application.streamingAssetsPath + "/" +             filPath + ".lua.txt";        return System.Text.Encoding.UTF8.GetBytes(File.ReadAllText(absPath));    }}

我们在这个myloader中得到我们要创建的文件的目录,可以通过以上的方法来得到。通过addloader这个方法加入回到函数,上面已经说过addloader可以注册回调,回调是字符串,该回调是一个byte数组。因为是byte数组,所以我们在传递之前就要将字符串改成字节数组才行,这样addloader才能编译。

二、C#访问Lua

1.访问一个全局的基本数据类型

这里指的是C#主动发起对Lua数据结构的访问。我们在将虚拟状态机创建好,可以通过luaenv.Global.Get<T>()这个方法来得到,所以我们可以写下面的代码,完成在C#中访问lua中的变量。在这里创建lua.txt就不再多说了,步骤是一样的。

void Start () {        LuaEnv luaenv = new LuaEnv();        //打开指定的文件夹        luaenv.DoString("require 'CSharpCallLua'");        int a = luaenv.Global.Get
("a"); string str = luaenv.Global.Get
("str"); bool isFly = luaenv.Global.Get
("isFly"); print(a + " " + str + " " + isFly); luaenv.Dispose(); }

2.访问一个全局的table

在c#中一共有4种方式访问lua中的table,在这里就不说哪几个,我们一个一个来说明。

  2.1映射到普通的class或struct

  定义一个class,有对应与table的字段的public属性,而且有无参数构造函数即可,比如定义一个包含public int f2;的class。这种方式下xlua会帮助你new一个实例,并把对应的字段赋值过去。table的属性可以多于或者少于class的属性。可以镶嵌其他复杂类型。要注意的是,这个过程比较复杂代价会比较大。数值也不会同步。

// Use this for initialization    void Start () {        LuaEnv luaenv = new LuaEnv();        //打开指定的文件夹        luaenv.DoString("require 'CSharpCallLua'");        //int a = luaenv.Global.Get
("a"); //string str = luaenv.Global.Get
("str"); //bool isFly = luaenv.Global.Get
("isFly"); //print(a + " " + str + " " + isFly); Person p = luaenv.Global.Get
("person"); print(p.name + " " + p.age); p.name = "dfdfdf"; luaenv.DoString("print(person.name)"); luaenv.Dispose(); } class Person { public string name;public int age; }

得到的效果如下:

我们从图中可以看见,我们在c#中改变person.name的值,输出时并不会发生改变,就像前面所说的一样,这个方式就是将lua中的值赋值下来,并不会改变lua中的值。这个方式我么首先应该在c#中创建一个与之对应的类Person,在这个类中有对应的值,然后通过方法,将lua中的变量映射到Person类中,这样就行了。关于映射,我们在高一学习函数的时候就学过了,函数是一对一,映射就是多对多的关系。其实我们也是没有必要这么在意映射的概念,可以理解成将lua中的变量给Person中对应的变量赋值。

 2.2映射到一个interface

  这种方式依赖与生成代码,代码生成器会生成这个interface的实例,如果get一个属性,生成代码会get对应的table字段,如果set属性也会设置对应的字段。甚至可以通过interface的方法访问lua的函数。

void Start () {        LuaEnv luaenv = new LuaEnv();        //打开指定的文件夹        luaenv.DoString("require 'CSharpCallLua'");        //int a = luaenv.Global.Get
("a"); //string str = luaenv.Global.Get
("str"); //bool isFly = luaenv.Global.Get
("isFly"); //print(a + " " + str + " " + isFly); //第一种方式访问 //Person p = luaenv.Global.Get
("person"); //print(p.name + " " + p.age); IPerson p = luaenv.Global.Get
("person"); print(p.name + " " + p.age); p.name = "dfdfdf"; luaenv.DoString("print(person.name)"); p.eat(3,4); luaenv.Dispose(); } //class Person { public string name;public int age; } [CSharpCallLua] interface IPerson { string name { get; set; } int age { get; set; }void eat(int a,int b); }

效果图如下:

从上面我们可以看见,我们通过这种方式改变lua中的值是可以改变成功的,因为这种方式实际上是一种引用的方式,在我们创建接口的时候一定要声明接口的类型,如上面的代码一样,否则就会报错。下面是lua的代码:

看到上面两种函数的区别的吗?没有:的话,就会需要加一个变量,其实这个变量你可以把它理解成从c#中的this,就是构造自己,在调用的时候。如果有:的话,就不用写这个变量了,因为lua会帮你构造的。好了,本节的内容就到这里了,拜拜。

转载于:https://www.cnblogs.com/jake-caiee/p/9818713.html

你可能感兴趣的文章
一次优化记录
查看>>
如何调用一个数据完整的firefox浏览器
查看>>
cgroup代码浅析(2)
查看>>
会计的思考(42):会计如何转变为公司的内部财务顾问
查看>>
利用钥匙串,在应用里保存用户密码的方法
查看>>
final,finally和finalize之间的区别
查看>>
python 装饰器
查看>>
[辟谣]下蹲猛起来眼前发黑是心脏衰竭的表现?别扯了!
查看>>
paper 96:计算机视觉-机器学习近年部分综述
查看>>
vuex状态管理详细使用方法
查看>>
不要等有了足够的钱才选择去创业!!!
查看>>
手把手教你画嘴巴,以后再也不怕画嘴巴了
查看>>
selenium - webdriver - 截图方法get_screenshot_as_file()
查看>>
linux 命令 — archive
查看>>
强大的jQuery网格插件 ParamQuery
查看>>
io.lettuce.core.RedisCommandTimeoutException: Command timed out
查看>>
种子填充算法描述及C++代码实现
查看>>
Kali渗透测试——快速查找Metasploit的模块
查看>>
如何生成项目的chm文档
查看>>
java封装httpClient工具(支持http和https,包含get和post请求)
查看>>