当前位置:   article > 正文

lua游戏服务器热更新_lua热更函数但不修改变量

lua热更函数但不修改变量

Lua热更新实现

用途

在生产环境上,总有可能出现不可预知的Bug,而通常修改好Bug仅仅又修改几句,停机维护的成本又太高,对于游戏来说,通常每个服就是单独的进程,也做不到像分布式环境下,关掉一部分机器,先升级一部分,再升级另一部分的无缝升级。这时候如果有热更就可以迅速的把Bug修复方案通过热更新进行修复,不会对用户任何的影响。例如:

  1. 业务逻辑有Bug
  2. 配置的数据有误
  3. 需求发生变更

热更新的原则

1、热更新不破坏原有数据

热更新更新的基本内容就是更新服务的逻辑,通常只是逻辑发生变化,但原有的值并不能被改变,例如:

  1. local a = 1
  2. function get_a()
  3. return a
  4. end

此时,我们调用get_a()返回是的1,我们将热更成

  1. local a = 2
  2. function get_a()
  3. print("get_a function")
  4. return a
  5. end

此时我们改变了a的初始值,但我们并不知道之前服务a的值是不是被重新赋过值,假设热更前a的值仍然为1,那么我们热更后调用get_a()返回的应该是1,而不应受新的初始值影响,而且同能打印出了"get_a function",这时候则认为热更正常。

2、不为热更新写更多的代码

热更新可以通过很多种方法实现,比如说模块为了支持数据不变的特性,需要在模块里额外写一些代码来记录旧值,热更新之后再把旧值copy过来,或者用一些特殊的语法来支撑。这种方法将会对项目增加很多的负担,而且一旦发生意料之外的Bug,热更系统几乎处于半瘫痪状态。应该来说,代码原本该怎么实现就怎么实现,对于99%的lua代码都是支持的,不需要修改来迎合热更新。通常热更新不改变原有变量值的类型。

热更新的实现,代码适用于5.2以上

原理

利用_ENV环境,在加载的时候把数据加载到_ENV下,然后再通过对比的方式修改_G底下的值,从而实现热更新,函数

function hotfix(chunk, check_name)

定义env的table,并为env设置_G访问权限,然后调用load实现把数据重新加载进来

  1. local env = {}
  2. setmetatable(env, { __index = _G })
  3. local _ENV = env
  4. local f, err = load(chunk, check_name, 't', env)
  5. assert(f,err)
  6. local ok, err = pcall(f)
  7. assert(ok,err)

此时env我们可以得到新函数有变更的部分,我们替换的为可见变量,也就是可直接访问的变量

  1. for name,value in pairs(env) do
  2. local g_value = _G[name]
  3. if type(g_value) ~= type(value) then
  4. _G[name] = value
  5. elseif type(value) == 'function' then
  6. update_func(value, g_value, name, 'G'..' ')
  7. _G[name] = value
  8. elseif type(value) == 'table' then
  9. update_table(value, g_value, name, 'G'..' ')
  10. end
  11. end

通过env当前的值和_G当前的值进行对比

  1. 如果类型不同我们直接覆盖原值,此时value不为nil,不会出现原则被覆盖成nil的情况
  2. 如果当前值为函数,我们进行函数的upvalue值比对
    1. function update_func(env_f, g_f, name, deep)
    2. --取得原值所有的upvalue,保存起来
    3. local old_upvalue_map = {}
    4. for i = 1, math.huge do
    5. local name, value = debug.getupvalue(g_f, i)
    6. if not name then break end
    7. old_upvalue_map[name] = value
    8. end
    9. --遍历所有新的upvalue,根据名字和原值对比,如果原值不存在则进行跳过,如果为其它值则进行遍历env类似的步骤
    10. for i = 1, math.huge do
    11. local name, value = debug.getupvalue(env_f, i)
    12. if not name then break end
    13. local old_value = old_upvalue_map[name]
    14. if old_value then
    15. if type(old_value) ~= type(value) then
    16. debug.setupvalue(env_f, i, old_value)
    17. elseif type(old_value) == 'function' then
    18. update_func(value, old_value, name, deep..' '..name..' ')
    19. elseif type(old_value) == 'table' then
    20. update_table(value, old_value, name, deep..' '..name..' ')
    21. debug.setupvalue(env_f, i, old_value)
    22. else
    23. debug.setupvalue(env_f, i, old_value)
    24. end
    25. end
    26. end
    27. end
  3. 如果当前值为table,我们遍历table值进行对比
    1. local protection = {
    2. setmetatable = true,
    3. pairs = true,
    4. ipairs = true,
    5. next = true,
    6. require = true,
    7. _ENV = true,
    8. }
    9. --防止重复的table替换,造成死循环
    10. local visited_sig = {}
    11. function update_table(env_t, g_t, name, deep)
    12. --对某些关键函数不进行比对
    13. if protection[env_t] or protection[g_t] then return end
    14. --如果原值与当前值内存一致,值一样不进行对比
    15. if env_t == g_t then return end
    16. local signature = tostring(g_t)..tostring(env_t)
    17. if visited_sig[signature] then return end
    18. visited_sig[signature] = true
    19. --遍历对比值,如进行遍历env类似的步骤
    20. for name, value in pairs(env_t) do
    21. local old_value = g_t[name]
    22. if type(value) == type(old_value) then
    23. if type(value) == 'function' then
    24. update_func(value, old_value, name, deep..' '..name..' ')
    25. g_t[name] = value
    26. elseif type(value) == 'table' then
    27. update_table(value, old_value, name, deep..' '..name..' ')
    28. end
    29. else
    30. g_t[name] = value
    31. end
    32. end
    33. --遍历table的元表,进行对比
    34. local old_meta = debug.getmetatable(g_t)
    35. local new_meta = debug.getmetatable(env_t)
    36. if type(old_meta) == 'table' and type(new_meta) == 'table' then
    37. update_table(new_meta, old_meta, name..'s Meta', deep..' '..name..'s Meta'..' ' )
    38. end
    39. end
更新

1、可以调用hotfix_file对整个文件进行热更

  1. function hotfix_file(name)
  2. local file_str
  3. local fp = io.open(name)
  4. if fp then
  5. io.input(name)
  6. file_str = io.read('*all')
  7. io.close(fp)
  8. end
  9. if not file_str then
  10. return -1
  11. end
  12. return hotfix(file_str, name)
  13. end

2、可以通过hotfix进行代码的更新

function hotfix(chunk, check_name)
关于坑

这里有一个注意事项,lua的module模块,如:

module("AA", package.seeall)

当我们加载lua模块的时候,这时候这个模块信息并不像初始化全局代码一样,就算提前设置了package.loaded["AA"] = nil, 也不会出现在env中同时也不会调用_G的__newindex函数,也就是说env["AA"]为空,故这种写法无法进行热更新,所以通常模块的写法改成如下

  1. --定义模块AA
  2. AA = {}
  3. --相当于package.seeall
  4. setmetatable(AA, {__index = _G})
  5. --环境隔离
  6. local _ENV = AA
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Gausst松鼠会/article/detail/189532
推荐阅读
相关标签
  

闽ICP备14008679号