终于我们把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方法热更新 lua为什么能热更新_c#

 

其实这个算法也是没简单的,只要将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<TextAsset>("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<int>("a");
        string str = luaenv.Global.Get<string>("str");
        bool isFly = luaenv.Global.Get<bool>("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<int>("a");
        //string str = luaenv.Global.Get<string>("str");
        //bool isFly = luaenv.Global.Get<bool>("isFly");
        //print(a + "  " + str + "   " + isFly);
        Person p = luaenv.Global.Get<Person>("person");
        print(p.name + "   " + p.age);
        p.name = "dfdfdf";
        luaenv.DoString("print(person.name)");
        luaenv.Dispose();
    }

    class Person { public string name;public int age; }

得到的效果如下:

lua方法热更新 lua为什么能热更新_lua_02

我们从图中可以看见,我们在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<int>("a");
        //string str = luaenv.Global.Get<string>("str");
        //bool isFly = luaenv.Global.Get<bool>("isFly");
        //print(a + "  " + str + "   " + isFly);

        //第一种方式访问
        //Person p = luaenv.Global.Get<Person>("person");
        //print(p.name + "   " + p.age);
        IPerson p = luaenv.Global.Get<IPerson>("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为什么能热更新_字符串_03

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

lua方法热更新 lua为什么能热更新_字符串_04

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