• <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 戰魂小筑 閱讀(8554) 評論(0)  編輯 收藏 引用 所屬分類: 游戲開發技術網絡 服務器技術Golang
            久久福利青草精品资源站| 久久精品无码一区二区日韩AV| 久久精品免费全国观看国产| 久久青青色综合| 久久精品免费观看| 亚洲&#228;v永久无码精品天堂久久 | 亚洲国产精品热久久| 久久激情五月丁香伊人| 日韩va亚洲va欧美va久久| 99久久国产亚洲综合精品| 久久综合九色综合欧美狠狠| 奇米影视7777久久精品人人爽| 天天综合久久久网| 久久丫精品国产亚洲av不卡| 久久综合一区二区无码| 精品久久久久久国产91| 99精品国产综合久久久久五月天| 国产精品免费久久久久久久久| 亚洲色欲久久久综合网东京热| 久久青青草原精品国产不卡| 国产精品久久久久…| 久久影院综合精品| 怡红院日本一道日本久久 | 99久久伊人精品综合观看| 中文字幕乱码久久午夜| 无码精品久久一区二区三区| 国产精品青草久久久久福利99 | 国产午夜福利精品久久2021| 亚洲中文字幕无码久久精品1| 亚洲精品无码久久毛片| 国产精品欧美久久久久无广告| 精品国产乱码久久久久久郑州公司| 久久人与动人物a级毛片| 免费精品久久久久久中文字幕| 久久久久无码专区亚洲av| 国产精品成人久久久久三级午夜电影 | 伊人情人综合成人久久网小说| 久久精品不卡| 久久强奷乱码老熟女网站| 污污内射久久一区二区欧美日韩| 久久无码一区二区三区少妇|