• <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
            久久久久无码精品国产app| 狠狠综合久久综合88亚洲| 亚洲第一极品精品无码久久| 久久综合狠狠综合久久综合88| 久久99精品国产一区二区三区| 一本大道久久a久久精品综合| 精品久久久久久99人妻| 久久精品国产亚洲AV无码娇色| 品成人欧美大片久久国产欧美... 品成人欧美大片久久国产欧美 | 中文国产成人精品久久亚洲精品AⅤ无码精品 | 亚洲国产精品成人久久蜜臀| 久久精品一区二区| 香蕉99久久国产综合精品宅男自| 久久精品人人槡人妻人人玩AV | 亚洲中文精品久久久久久不卡 | 久久国产免费直播| 国产精品美女久久久久av爽| 麻豆av久久av盛宴av| 人人狠狠综合久久亚洲高清| 久久久国产乱子伦精品作者| 久久免费观看视频| 亚洲伊人久久成综合人影院 | 一级做a爰片久久毛片免费陪 | 久久久久AV综合网成人| 亚洲精品WWW久久久久久| 亚洲嫩草影院久久精品| 久久丫精品国产亚洲av不卡| 久久精品免费全国观看国产| 久久亚洲AV无码精品色午夜| 中文字幕亚洲综合久久2| 久久99精品久久久久久久久久| 热re99久久精品国99热| 久久婷婷色综合一区二区| 国产福利电影一区二区三区久久久久成人精品综合 | 久久99国产综合精品女同| 性做久久久久久久久老女人| 成人亚洲欧美久久久久| 国产成人久久激情91| 国产精品丝袜久久久久久不卡| 久久久久亚洲精品天堂| 亚洲国产另类久久久精品黑人|