1. 程式人生 > >lua遊戲伺服器熱更新

lua遊戲伺服器熱更新

Lua熱更新實現

用途

在生產環境上,總有可能出現不可預知的Bug,而通常修改好Bug僅僅又修改幾句,停機維護的成本又太高,對於遊戲來說,通常每個服就是單獨的程序,也做不到像分散式環境下,關掉一部分機器,先升級一部分,再升級另一部分的無縫升級。這時候如果有熱更就可以迅速的把Bug修復方案通過熱更新進行修復,不會對使用者任何的影響。例如:

  1. 業務邏輯有Bug
  2. 配置的資料有誤
  3. 需求發生變更

熱更新的原則

1、熱更新不破壞原有資料

熱更新更新的基本內容就是更新服務的邏輯,通常只是邏輯發生變化,但原有的值並不能被改變,例如:

local a = 1
functionget_a()
    return
a end

此時,我們呼叫get_a()返回是的1,我們將熱更成

local a = 2
functionget_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底下的值,從而實現熱更新,函式

functionhotfix(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) do
    local g_value = _G[name]
    if type(g_value) ~= type(value) then
        _G[name] = value
    elseif type(value) == 'function' then
        update_func(value, g_value, name, 'G'..'  ')
        _G[name] = value
    elseif type(value) == 'table' then
        update_table(value, g_value, name, 'G'..'  ')
    end
end

通過env當前的值和_G當前的值進行對比

  1. 如果型別不同我們直接覆蓋原值,此時value不為nil,不會出現原則被覆蓋成nil的情況
  2. 如果當前值為函式,我們進行函式的upvalue值比對
    functionupdate_func(env_f, g_f, name, deep)
     --取得原值所有的upvalue,儲存起來
     local old_upvalue_map = {}
     for i = 1, math.huge do
         local name, value = debug.getupvalue(g_f, i)
         if not name then break end
         old_upvalue_map[name] = value
     end
     --遍歷所有新的upvalue,根據名字和原值對比,如果原值不存在則進行跳過,如果為其它值則進行遍歷env類似的步驟
     for i = 1, math.huge do
         local name, value = debug.getupvalue(env_f, i)
         if not name then break end
         local old_value = old_upvalue_map[name]
         if old_value then
             if type(old_value) ~= type(value) then
                 debug.setupvalue(env_f, i, old_value)
             elseif type(old_value) == 'function' then
                 update_func(value, old_value, name, deep..'  '..name..'  ')
             elseif type(old_value) == 'table' then
                 update_table(value, old_value, name, deep..'  '..name..'  ')
                 debug.setupvalue(env_f, i, old_value)
             else
                 debug.setupvalue(env_f, i, old_value)
             end
         end
     end
    end
    
  3. 如果當前值為table,我們遍歷table值進行對比
    local protection = {
     setmetatable = true,
     pairs = true,
     ipairs = true,
     next = true,
     require = true,
     _ENV = true,
    }
    --防止重複的table替換,造成死迴圈
    local visited_sig = {}
    functionupdate_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 end
     local signature = tostring(g_t)..tostring(env_t)
     if visited_sig[signature] then return end
     visited_sig[signature] = true
     --遍歷對比值,如進行遍歷env類似的步驟
     for name, value in pairs(env_t) do
         local old_value = g_t[name]
         if type(value) == type(old_value) then
             if type(value) == 'function' then
                 update_func(value, old_value, name, deep..'  '..name..'  ')
                 g_t[name] = value
             elseif type(value) == 'table' then
                 update_table(value, old_value, name, deep..'  '..name..'  ')
             end
         else
             g_t[name] = value
         end
     end
     --遍歷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' then
         update_table(new_meta, old_meta, name..'s Meta', deep..'  '..name..'s Meta'..'  ' )
     end
    end
    

更新

1、可以呼叫hotfix_file對整個檔案進行熱更

functionhotfix_file(name)
    local file_str
    local fp = io.open(name)
    if fp then
        io.input(name)
        file_str = io.read('*all')
        io.close(fp)
    end

    if not file_str then
        return -1
    end
    return hotfix(file_str, name)
end

2、可以通過hotfix進行程式碼的更新

functionhotfix(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遊戲伺服器更新

Lua熱更新實現 用途 在生產環境上,總有可能出現不可預知的Bug,而通常修改好Bug僅僅又修改幾句,停機維護的成本又太高,對於遊戲來說,通常每個服就是單獨的程序,也做不到像分散式環境下,關掉一部分機器,先升級一部分,再升級另一部分的無縫升級。這時候如果有熱更就可以迅

專屬妹子開發之AssetBundles伺服器更新

using System.Collections; using System.Collections.Generic; using UnityEngine; using System.IO; using System; using UnityEngine.UI; using UnityEngine.

一步一步開發Game伺服器(三)載入指令碼和伺服器更新

大家可能對遊戲伺服器的執行不太理解或者說不太清楚一些機制。 但是大家一定會明白一點,當程式在執行的時候出現一些bug,必須及時更新,但是不能重啟程式的情況下。 這裡牽涉到一個問題。比如說在遊戲裡面,,如果一旦開服,錯非完全致命性bug,否則是不能頻繁重啟伺服器程式的, 你重啟一次就可能流失一部分玩家。那

一步一步開發Game伺服器(三)載入指令碼和伺服器更新(二)完整版

可是在使用過程中,也許有很多會發現,動態載入dll其實不方便,應為需要預先編譯程式碼為dll檔案。便利性不是很高。 那麼有麼有辦法能做到動態實時更新呢???? 官方提供了這兩個物件,動態編譯原始檔。 提供對 C# 程式碼生成器和程式碼編譯器的例項的訪問。 CSharpCodeProvider

伺服器更新(當個檔案更新

上一篇文章我介紹瞭如果動態載入dll檔案來更新程式 可是在使用過程中,也許有很多會發現,動態載入dll其實不方便,應為需要預先編譯程式碼為dll檔案。便利性不是很高。 那麼有麼有辦法能做到動態實時更新呢???? 官方提供了這兩個物件,動態編譯原始檔。 提供對 C# 程式碼生成器和程式碼編譯器的例項的訪

Tomcat JRrebel遠端伺服器更新

環境:eclipse衍生IDE,我這邊用的是sts。Linux遠端伺服器,Tomcat。 1、安裝外掛  Help > Install New Software   2、啟用 註冊賬號,Help > JRebel > Activation啟用JRe

伺服器更新的討論

我們之前的伺服器是多程序純C#架構的伺服器。最近遊戲上線,遇到有時候需要線上修bug的問題。之前對程式碼熱更新預料不足,導致在線上出了一些bug時非常被動,往往需要重啟伺服器解決問題,影響前期的體驗。我們的遊戲型別是一個RPG卡牌遊戲,操作偏向於單機向,大量依賴於廣播的操作比

Unity使用XLua進行遊戲更新

XLua匯入Unity工程 另外匯入Xlua中的 Toos資料夾複製到unity工程的Asset同級 1開啟巨集 HOTFIX_ENABLE 注意: Unity的程式碼改變後要執行下面兩個步驟  要確定沒報錯 選擇該專案升級為7.0語言 還是報錯

Unity伺服器更新專案總結

1.       當下載unity3D、txt、xml檔案時,切記在IIS中新增相應的MIME型別,否則會出現400錯誤 2.      www、bundleAsset、Asset之間的關係: 1.              WebStream:包括了壓縮的檔案,解壓所需的

C++結合LUA實現程式碼更新

    最近寫一個檔案伺服器,是用C++寫的,功能基本都實現了,老大提到一個問題就是如果要往伺服器增加一個功能,那就得重新編譯程式,對於伺服器這7*24的執行程式是行不通的. 所以自己結合指令碼語言LUA實現了一個簡單的熱更新模板. 程式是這樣的. 主程式用C++實現,負責

從nginx更新聊一聊Golang中的伺服器更新(上)

從nginx熱更新聊一聊Golang中的熱更新(上) 靜態語言在伺服器程式設計時都會遇到這樣的問題:如何保證已有的連線服務不中斷同時又升級版本? 最近花了點時間看了下nginx熱更新程式碼流程,想了下結合之前的經驗一併總結下熱更新 熱更新是什麼? 舉個例子,

更新-----為何使用lua進行

  事實上我們在安卓端是可以使用c#jit的,但是我們在ios上的程式碼是AOT(預先編譯,靜態編譯)的,不能用c# jit(實時編譯,即時編譯)。   ios不能用c#熱更是因為啟動了CPU的No eXecute bit,簡單說就是ios不允許更改程式碼,更新程式碼必須重新提安裝包進行稽核,AppStor

iOS 中 lua 更新 'system' is unavailable: not available on iOS

iOS熱更新-8種實現方式 一、JSPatch 熱更新時,從伺服器拉去js指令碼。理論上可以修改和新建所有的模組,但是不建議這樣做。 建議 用來做緊急的小需求和 修復嚴重的線上bug。 二、lua指令碼 比如: wax。熱更新時,從伺服器拉去lua指令碼。遊戲開發經常用到。

013-更新lua語言六

學習到現在我們對lua也是具有很深的瞭解,知道lua是熱更新的重要語言,所以我們要繼續學習lua。今天我們學習的是lua檔案 Lua檔案 I/O 其實這個都是我們是很熟悉的,它的內容也沒有多少。記得我們在考江蘇二級的時候,最後一題總是會考這個,其中最有名的是printf與fprintf了。只要知道一個是往

ToLua更新之LuaFramework框架之編寫Lua邏輯(3)

1、Lua的Update方法 第一篇“程式碼熱更新”演示了用lua列印HelloWorld的方法,第二篇“資源熱更新”演示了載入坦克模型的方法。這一篇要把兩者結合起來,用lua實現“用鍵盤控制坦克移動”的功能。用Lua和用c#編寫的Unity3D程式大同小異,只需正確使用A

Unity 伺服器踩坑 Node.js 與 Express 資源更新與檔案傳輸(三)

一、下載安裝Node.js 安裝,傻瓜式點選教程 二、安裝Express 開啟cmd一步一步cd進入C:\Program Files\nodejs\node_modules\npm資料夾下 然後執行npm install --sav

Cocos Creator 集合類遊戲模組化與更新實踐

集合類遊戲常見於大廳+子游戲模式的棋牌類專案,這類專案通常是由一系列小專案模組組合而成的系統綜合工程。基特點是模組開發週期短,模組版本迭代快,維護週期長,除了各遊戲模組本身玩法邏輯,還需要面對眾多模組的管理任務。 集合類遊戲從表面上看似簡單,但在模組設計、資源管理、多人協作上需要挑戰的問題不少,專

實現 React Naitve 更新 (client && server) 客戶端以及伺服器

目前針對react native 熱更新的方案比較成熟的選擇有microsoft公司的code-push 傳送門,與react-native 中文網的pushy 傳送門 本文選擇code-push 用來進行對react-native 實現熱更新,code-pus

騰訊開源手遊更新方案Xlua嚐鮮(三)——C#訪問Lua

C#訪問Lua 這裡指的是C#主動發起對Lua資料結構的訪問。 一、獲取一個全域性基本資料型別 訪問LuaEnv.Global就可以了,上面有個模版Get方法,可指定返回的型別。 luaenv.Global.Get<int>("a"); luaenv.Globa

遊戲更新雜談

熱更新的內容可以是美術資源, 可以是程式碼, 但相對來說, 美術資源的更新不會受到約束, 程式碼實際上是重災區, 本文介紹的主要是程式碼熱更新 熱更新對於開發者來說是一件麻煩事, 特別對於看重效率,便捷性和結構的程式設計師來說, 熱更新就是運營人員的不懂技術的表現然而, 對於上線才是剛剛開始的網路遊戲