脚本的热更新的流程都大同小异, 第一步先保存旧代码的块的数据, 第二部加载新的代码块,第三步将旧代码块的局部和全局数据拷贝到新代码块的对应的
变量中。
在服务器热更新中,主要考虑热更的内容是什么, 一般更新分两种,一种是逻辑的更新,一种是变量的值更新。
先谈论第一种,在lua 5.3中,继承了新的debug.upvaluejoin(f1, i, f2, j)函数,能将函数f1的第i个个upvalue引用f2的第j个upvalue,
可以通过debug.getupvalue函数,获取ENV 的upvalue来的索引位置,找到该位置后,可以通过debug.upvaluejoin函数来实现env的交换。
第二种就是要更新变量值,lua中的变量值更新,除了table外,其它变量类型都可以通过直接赋值来改变值的大小。由于table的回收,会导致其它引用
该table的变量都会收到影响,会导致错误的发生,所以, table的值更新,不能通过直接赋值, 可以通过移除table里面的所有元素的值,table的table除外,
然后将新table的所有元素的值遍历赋值给旧的table,注意的是不能将table也赋值,如果新table里有新的子元素是table类型,必须要在旧table对应值里重新
创建一个新的table,递归将值拷贝。
以下是table的拷贝方法实现热更的过程:
HotFixTable = {} -- copy a old table's all key and value to new table, but don't release old table local function copyTable( oldTbl, newTbl) for k, v in pairs(newTbl) do if type(v) == "table" then if type(oldTbl[k]) ~= "table" then oldTbl[k] = {} end copyTable( oldTbl[k], v) else oldTbl[k] = v end end return oldTbl, newTbl end --table no need release local function clearTable(tbl) for k, v in pairs(tbl) do if type(v) ~= "table" then tbl[k] = nil else clearTable(v) end end return tbl end --use file to hotfix the source table function HotFixTable.hotFixByFileName(fileName, sourceTbl) local deskTbl local function newIndexFunc(tbl, name, value)--t:table, name:key, f:function deskTbl = value end local dummy_env = setmetatable({}, { __newindex = newIndexFunc }) local f, err = loadfile(fileName, "bt", dummy_env) if f then local ok, err = pcall(f) assert(ok, err) if ok then clearTable(sourceTbl) copyTable(sourceTbl, deskTbl) end end end --use lua string to hotfix the table function HotFixTable.hotFixByData(hitfixContent, sourceTbl) local deskTbl local function newIndexFunc(tbl, name, value)--t:table, name:key, f:function deskTbl = value end local dummy_env = setmetatable({}, { __newindex = newIndexFunc }) local f, err = load(hitfixContent, "=patch", "bt", dummy_env) if f then local ok, err = pcall(f) assert(ok, err) if ok then clearTable(sourceTbl) copyTable(sourceTbl, deskTbl) end end end
调用方法如下:
HotFixTable.hotFixByFileName("新table的保存所在文件", 旧table)