• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            戰魂小筑

            討論群:309800774 知乎關注:http://zhihu.com/people/sunicdavy 開源項目:https://github.com/davyxu

               :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理 ::
              257 隨筆 :: 0 文章 :: 506 評論 :: 0 Trackbacks

            本文主要研究游戲服務器帶狀態的熱更新需求 http的無狀態熱更新需求已經有成熟方案, 故不在本文描述范圍

            基本概念

            • Golang的熱更新采用什么機制?

              使用go1.8提供的plugin包機制實現

            • plugin包本身設計的目的是熱更新么?

              plugin包其實只是支持將代碼分別編譯為多個動態庫,動態加載后運行 并不能完全支持類似C/C++的動態庫方式處理代碼

            • 帶狀態的進程熱更新的基本概念及范圍是什么?

              數據部分(model)不更新, 只更新邏輯部分(函數)

            • 表格和配置更新算熱更新么?

              算, 但不是本文描述范圍

            • 熱更新能在windows上使用么?

              不支持

            代碼及結構

            • 我能將原來一個exe的代碼編譯為so提供給plugin使用么?

              可以, 但是必須仍然保留main包作為插件入口, 并在main包內添加提供給plugin調用入口.

            • 如果動態庫中沒有main包, 編譯出的so能用么?

              不能, 包必須包含main, 否則輸出的是.a的文件, plugin包加載會報錯

            • 動態庫中, 非main包的的代碼修改能做熱更新么?

              不能!(崩潰了吧, 我提了一個issue: https://github.com/golang/go/issues/20554)

              如果確實做了修改, 底層會報錯: plugin was built with a different version of package

              解決方法: 修改plugin包底層實現并重新編譯 打開runtime/plugin.go, 注釋以下代碼 for _, pkghash := range md.pkghashes { if pkghash.linktimehash != *pkghash.runtimehash { return "", nil, pkghash.modulename } } 執行/usr/local/go/run.bash 重編譯+測試

            • 代碼中哪些可以被更新? 方法可以被更新么? 閉包呢?

              只能更新擁有靜態地址的結構.例如: 包級別函數(類似于靜態函數)

              例如: svc_0.so里有一個Foo函數, svc_1.so修改了Foo函數實現, 熱更新可實現

              閉包=函數+捕獲變量, 實際上是一個動態結構, 沒有靜態地址, 無法被更新

              各種包級別變量, 結構體變量, 結構體方法, 局部變量均不能被熱更新, 但是變量值不會被影響

              新增結構可以被運行

            • 使用結構體方法調用了包級別函數, 包級別函數能被更新么?

              可以, 雖然方法不能被更新, 但方法被調用的包級別函數的地址是固定的, 所以可以被熱更新

            • init包初始化函數能用么? 能被熱更新么?

              官方這樣描述:

              When a plugin is first opened, the init functions of all packages not already part of the program are called. The main function is not run. A plugin is only initialized once, and cannot be closed

              插件第一次被打開時, 其關聯的, 沒有成為程序的一部分的所有的包的init函數將被調用. 插件的main函數不會被調用. 插件只會被初始化一次, 不能被關閉

              因此, 需要手動將init函數改成自己的函數, 統一在so的main包里調用

            編譯

            • 如何編譯獲得plugin包支持的動態庫

              SVCNAME=$1 SVCVER=$2 TIMESTAMP=`date '+%Y%m%d_%H%M%S'` go build -v -buildmode=plugin --ldflags="-pluginpath=${SVCNAME}_${TIMESTAMP}" -o ${SVCNAME}_${SVCVER}.so ${SVCNAME}

              -buildmode=plugin是重要參數

              --ldflags里的-pluginpath的作用是: 每次編譯的內部識別路徑都是不同的, 避免重復加載的警告

              參考: https://github.com/golang/go/issues/19004

            posted on 2017-07-06 12:47 戰魂小筑 閱讀(8537) 評論(0)  編輯 收藏 引用 所屬分類: 游戲開發技術網絡 服務器技術Golang
            亚洲另类欧美综合久久图片区| 久久激情五月丁香伊人| 国产精品久久成人影院| 久久亚洲AV成人无码电影| 久久福利青草精品资源站免费| 久久免费99精品国产自在现线| 一本色综合网久久| 99久久精品国产麻豆| 久久精品中文字幕久久| 青青草原精品99久久精品66| 97久久精品午夜一区二区| 亚洲精品99久久久久中文字幕| 亚洲国产精品高清久久久| 久久免费大片| 久久99国内精品自在现线| 日日躁夜夜躁狠狠久久AV| 老司机午夜网站国内精品久久久久久久久 | 日韩久久久久中文字幕人妻| 99麻豆久久久国产精品免费| 看全色黄大色大片免费久久久| jizzjizz国产精品久久| 精品伊人久久大线蕉色首页| 色偷偷88欧美精品久久久| 精品免费tv久久久久久久| 看久久久久久a级毛片| 中文字幕亚洲综合久久菠萝蜜| 99久久婷婷国产一区二区| 伊人久久综在合线亚洲2019| 亚洲av日韩精品久久久久久a | 日日狠狠久久偷偷色综合96蜜桃 | 香蕉久久av一区二区三区| 久久久久久精品免费免费自慰| 久久精品亚洲乱码伦伦中文| 国产亚州精品女人久久久久久| 精品久久人人妻人人做精品| 国产精品久久久久久一区二区三区| 伊人久久综合无码成人网| 久久精品中文字幕大胸| 国色天香久久久久久久小说| 久久香蕉国产线看观看猫咪?v| 久久婷婷五月综合成人D啪|