• <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
            国内精品久久久久久久97牛牛| 久久综合五月丁香久久激情| 久久ZYZ资源站无码中文动漫| 久久精品国产亚洲av麻豆图片| 亚洲国产精品无码久久一区二区| 无码人妻久久一区二区三区免费丨| 久久99国产综合精品女同| 狠狠久久综合| 91精品国产综合久久久久久| 欧美久久综合九色综合| 久久久久亚洲av无码专区喷水| 久久国产三级无码一区二区| 午夜精品久久久久久毛片| 色偷偷91久久综合噜噜噜噜| av无码久久久久不卡免费网站 | 久久国产成人午夜aⅴ影院| 久久人人爽人人人人爽AV| 国内精品久久久久久久久| 久久99国产综合精品| 亚洲精品国精品久久99热一| 亚洲国产精品综合久久一线| 久久久久一区二区三区| 伊人久久无码中文字幕| 人人狠狠综合88综合久久| 久久久精品免费国产四虎| 日韩精品无码久久久久久| 香蕉aa三级久久毛片| 久久人搡人人玩人妻精品首页| 久久综合综合久久97色| 97久久精品无码一区二区天美| 久久人人爽爽爽人久久久| 午夜视频久久久久一区| 久久久青草青青国产亚洲免观| 国产精品无码久久综合网| 国产精品日韩深夜福利久久| 国产精品免费久久| 久久久免费观成人影院| 亚洲欧美日韩精品久久亚洲区| 久久这里只精品99re66| 国产精品一区二区久久精品涩爱| 色妞色综合久久夜夜|