Lua热更新实现

用途

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

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

热更新的原则

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

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

local a = 1
function get_a()return a
end

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

local a = 2
function get_a()print("get_a function")return a
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实现把数据重新加载进来

local env = {}
setmetatable(env, { __index = _G })
local _ENV = env
local f, err = load(chunk, check_name,  't', env)
assert(f,err)
local ok, err = pcall(f)
assert(ok,err)

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

for name,value in pairs(env) dolocal g_value = _G[name]if type(g_value) ~= type(value) then_G[name] = valueelseif type(value) == 'function' thenupdate_func(value, g_value, name, 'G'..'  ')_G[name] = valueelseif type(value) == 'table' thenupdate_table(value, g_value, name, 'G'..'  ')end
end

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

  1. 如果类型不同我们直接覆盖原值,此时value不为nil,不会出现原则被覆盖成nil的情况
  2. 如果当前值为函数,我们进行函数的upvalue值比对
    function update_func(env_f, g_f, name, deep)--取得原值所有的upvalue,保存起来local old_upvalue_map = {}for i = 1, math.huge dolocal name, value = debug.getupvalue(g_f, i)if not name then break endold_upvalue_map[name] = valueend--遍历所有新的upvalue,根据名字和原值对比,如果原值不存在则进行跳过,如果为其它值则进行遍历env类似的步骤for i = 1, math.huge dolocal name, value = debug.getupvalue(env_f, i)if not name then break endlocal old_value = old_upvalue_map[name]if old_value thenif type(old_value) ~= type(value) thendebug.setupvalue(env_f, i, old_value)elseif type(old_value) == 'function' thenupdate_func(value, old_value, name, deep..'  '..name..'  ')elseif type(old_value) == 'table' thenupdate_table(value, old_value, name, deep..'  '..name..'  ')debug.setupvalue(env_f, i, old_value)elsedebug.setupvalue(env_f, i, old_value)endendend
    end
    
  3. 如果当前值为table,我们遍历table值进行对比
    local protection = {setmetatable = true,pairs = true,ipairs = true,next = true,require = true,_ENV = true,
    }
    --防止重复的table替换,造成死循环
    local visited_sig = {}
    function update_table(env_t, g_t, name, deep)--对某些关键函数不进行比对if protection[env_t] or protection[g_t] then return end--如果原值与当前值内存一致,值一样不进行对比if env_t == g_t then return endlocal signature = tostring(g_t)..tostring(env_t)if visited_sig[signature] then return endvisited_sig[signature] = true--遍历对比值,如进行遍历env类似的步骤for name, value in pairs(env_t) dolocal old_value = g_t[name]if type(value) == type(old_value) thenif type(value) == 'function' thenupdate_func(value, old_value, name, deep..'  '..name..'  ')g_t[name] = valueelseif type(value) == 'table' thenupdate_table(value, old_value, name, deep..'  '..name..'  ')endelseg_t[name] = valueendend--遍历table的元表,进行对比local old_meta = debug.getmetatable(g_t)local new_meta = debug.getmetatable(env_t)if type(old_meta) == 'table' and type(new_meta) == 'table' thenupdate_table(new_meta, old_meta, name..'s Meta', deep..'  '..name..'s Meta'..'  ' )end
    end
    

更新

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

function hotfix_file(name)local file_strlocal fp = io.open(name)if fp thenio.input(name)file_str = io.read('*all')io.close(fp)endif not file_str thenreturn -1endreturn hotfix(file_str, name)
end

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

function hotfix(chunk, check_name)

关于坑

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

module("AA", package.seeall)

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

--定义模块AA
AA = {}
--相当于package.seeall
setmetatable(AA, {__index = _G})
--环境隔离
local _ENV = AA

lua游戏服务器热更新相关推荐

  1. 为什么游戏需要热更新?

    版权申明: 本文为"优梦创客"原创文章,您可以自由转载,但必须加入完整的版权声明 更多学习资源请加QQ:1517069595获取(企业级性能优化/热更新/Shader特效/服务器/ ...

  2. lua文件服务器,lua游戏服务器源码

    lua游戏服务器源码 内容精选 换一换 简要介绍Lua是一种轻量小巧的脚本语言,用标准C语言编写并以源代码形式开放, 其设计目的是为了嵌入应用程序中,从而为应用程序提供灵活的扩展和定制功能.语言:C一 ...

  3. 轩辕传奇服务器维护,《轩辕传奇》游戏服务器大更新

    [游戏介绍] <轩辕传奇>是以上古战争为核心的中式玄幻网游,采用新一代国际顶级引擎Gamebryo Element 2.3(XBOX360主要引擎),辅以丰富的动态效果和光影效果,带来呼之 ...

  4. 《天涯明月刀》游戏资源热更新解决方案

    天刀在韩国化的过程中,韩方对天刀的游戏内商城功能提出了一系列的适应他们本土运营习惯的商业化改造需求,其中最重要的基础功能修改就是需要商城能够支持不停服修改商品内容或上下架商品,包含新增未事先打进版本配 ...

  5. Parcel React 开发服务器热更新实战

    parcel Parcel 是 Web 应用打包工具,适用于经验不同的开发者.它利用多核处理提供了极快的速度,并且不需要任何配置. 内容 官网教程没有实现devSever和动态更新相结合具体部署步骤. ...

  6. 手游为什么要热更新,C#为什么不能热更新,LUA为什么可以

    手游为什么要热更新,C#为什么不能热更新,LUA为什么可以 热更新是什么?简单的说就是打补丁,只补需要部分,不用重个游戏包重打上传 热更新问题的本质是代码更新而不是资源更新,为什么呢? 大型手游都是将 ...

  7. [寒江孤叶丶的Cocos2d-x之旅_17]Cocos2d-x 3.2版本以上LUA脚本热更新(动态更新)解决方案

    原创文章,欢迎转载,转载请注明:文章来自[寒江孤叶丶的Cocos2d-x之旅系列] 博客地址:http://blog.csdn.net/qq446569365 能够进行热更新,是Lua脚本的最大优势, ...

  8. 【热更新】游戏热更新方案

    游戏热更新方案 热更新演化 热更新方案 [1] 进程切换 1.1 利用fork.exec切换 1.2 利用网关切换 1.3 微服务 - 进程切换注意要点 [2] 动态库替换 [3] 脚本语言热更新 热 ...

  9. 【游戏开发高阶】从零到一教你Unity使用ToLua实现热更新(含Demo工程 | LuaFramework | 增量 | HotUpdate)

    文章目录 零.前言 一.我做的热更新Demo 1.效果演示 2.流程图 3.工程源码 二.为什么要有热更新 三.Unity如何支持热更新 1.热更C#代码 2.热更lua代码与资源 四.Unity中集 ...

最新文章

  1. 高级持续性威胁检测无法检测出自定义恶意软件?
  2. mysql设置slave复制_mysql5.5建立主从复制(setupmaster-slavereplication)_MySQL
  3. matlab训练神经网络模型并导入simulink详细步骤
  4. git分散式版本管理系统,从安装到基本使用
  5. vue路由匹配实现包容性_我们甚至没有进行包容性的讨论
  6. Can‘t connect to MySQL server on ‘localhost‘(10061)【SQLyog】
  7. 模拟机安装linux教程,Windows 10利用虚拟机安装Linux图文教程
  8. 代码文档生成工具-Doxygen生成CHM和RTF图文教程
  9. U盘做启动盘后,如何恢复原始容量
  10. 知识竞赛现场管理系统安装配置及使用疑难问题汇编
  11. android手机鼠标,安卓手机变鼠标以及手机控制电脑图文教程
  12. linux 鼠标光标由箭头变成十字形恢复方法
  13. Depends简介与使用说明
  14. Tomcat9安装及配置步骤
  15. hill密码(希尔密码)
  16. python中英文切换_Python方法完成转换英文字符操作
  17. 请先切换至Wxml Pannel的解决方法
  18. 【已修复】Error: ValueError: The last dimension of the inputs to `Dense` should be defined. Found `None`
  19. 微信小程序网络请求异常怎么办_微信小程序网络超时的处理
  20. No module named gensim.corpora

热门文章

  1. Fireworks的常用快捷键
  2. android桌面文件夹,打造清新手机桌面 5款安卓桌面文件夹合辑推荐
  3. android之解析包时出现错误(二)
  4. UART串口协议基础1
  5. 反垃圾邮件网关该如何选型考量
  6. Java项目:养老院管理系统(java+Spring Boot + SpringMVC + MyBatis+HTML+CSS+JavaScrip+ Layui+maven+mysql)
  7. 尝试使用以5W1H分析法来学习5W1H分析法
  8. 今天是个好日子,让我们用Python采集今日全网热门话题评论,看看大家有多高兴
  9. 做互联网不要把自己看得太高
  10. C语言练习题,if-else的用法,体型判断:医务工作者经广泛的调查和统计分析,根据身高与体重因素给出了以下按“体指数”进行体型判断的方法