1.介绍

   使用脚本开发游戏业务逻辑其中一个好处就是代码可线上热更,不停机修复bug。而热更代码的写法与需要被热更的文件的代码又有着密切的关系,本文介绍一种热更方法。

2.热更原理

  Lua提供一个叫require的函数,实现文件的加载,require的作用可查看云风大大的参考手册说明  http://cloudwu.github.io/lua53doc/manual.html#pdf-require
要热更一个已经被加载(require)过的模块,要做两件事件.

1. 将package.loaded[filename] = nil, 将模块置空.

2. 重新调用require,require(filename).

如果要热更的模块里面所用到的都是全局变量,那么只需要这样写:

global_var = global_var

ok,新的模块内容就会重新被加载进内存了,而且全局变量也成功保留了热更前的值。

然而每个文件里面只定义全局变量,会破坏模块的可读性,试想一下,定义了一个类,但它并没有成员变量,用到的变量全是全局变量,这是一种怎样的感受。
所以我们要换一种写法,模块既可以有自己的变量,又可以被热更。

3. 实现

  形如一个类既有成员变量,又有成员函数,要实现热更一个模块同时又保留其原有变量,我们可以把模块的变量定义和函数定义分开,分别写在两个文件。
目录如下:

  data目录: modA.lua; modB.lua; modC.lua

logic目录: modA.lua; modB.lua; modC.lua

把模块的变量定义放在data文件夹,把模块函数的实现放在logic目录。

下面是实现:

 -- data.modA
local mod = {}
-- 这里定义模块变量
mod.players = {}
mod.total_online = -- 最后记得return这个表
return mod
 -- logic.lua
local mod = require("data.modA") -- 定义函数
function mod.login()
-- xxxxxx
end function mod.load_data()
-- xxxxxxx
end

这里只是一个例子,logic.modA想要执行 `require("data.modA")` 成功,还要修改package.path,这个是Lua虚拟机启动时自己去定义了,这里不做讨论。
这里把模块的变量和函数的定义分离了,我们只热更function目录下的文件,data目录下的则不管。
热更文件的实现如下:

 -- require_ex.lua
function require_ex(filename)
local old_content
if package.loaded[filename] then
-- 把旧的模块保存起来
old = package.loaded[filename]
-- 然后package.loaded[filename]赋空
package.loaded[filename] = nil
end -- xpcall下执行require
local ok,err = pcall(require, filename)
if not ok then
--热更失败,将旧值赋回去
print("hotfix fail, err msg ",err)
package.loaded[filename] = old_content
return false
end return true
end

切记: 重新require模块的时候,必须在pcall下执行,万一模块有语法错误,require挂掉的话,package.loaded[filename]就会一直为nil。当有其他文件需要执行 require(filename)的时候,则会报错,后面的逻辑无法执行。

04-24 05:25