• <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 戰魂小筑 閱讀(8576) 評論(0)  編輯 收藏 引用 所屬分類: 游戲開發技術網絡 服務器技術Golang
            久久ZYZ资源站无码中文动漫 | 欧美午夜精品久久久久免费视| 色偷偷88欧美精品久久久| 久久久国产亚洲精品| 久久66热人妻偷产精品9| 久久久久女教师免费一区| 人妻无码αv中文字幕久久琪琪布 人妻无码精品久久亚瑟影视 | 久久久久九九精品影院| 亚洲中文字幕久久精品无码喷水| 久久se精品一区精品二区| 一日本道伊人久久综合影| 欧美精品一区二区精品久久| 热久久视久久精品18| 国产福利电影一区二区三区久久老子无码午夜伦不 | 一本色道久久88精品综合| 国内精品久久久久久野外| 久久久久亚洲精品日久生情| 狠狠精品干练久久久无码中文字幕| 久久国产精品成人影院| 久久综合久久美利坚合众国| 久久久WWW免费人成精品| 国产一久久香蕉国产线看观看| 欧美日韩精品久久久免费观看| 国产精品99久久久久久猫咪| 精品国产91久久久久久久| 色欲av伊人久久大香线蕉影院| 久久夜色撩人精品国产小说| 人人狠狠综合久久亚洲婷婷| 久久99热狠狠色精品一区| 99久久99这里只有免费费精品| 亚洲日韩欧美一区久久久久我 | 人妻精品久久久久中文字幕69 | 国产精品成人99久久久久 | 国产成人精品综合久久久久| 少妇被又大又粗又爽毛片久久黑人| 国产精品成人精品久久久| 久久黄色视频| 伊人情人综合成人久久网小说| 久久伊人中文无码| 亚洲伊人久久精品影院| 伊人久久综合成人网|