• <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>
            Fork me on GitHub
            隨筆 - 215  文章 - 13  trackbacks - 0
            <2016年9月>
            28293031123
            45678910
            11121314151617
            18192021222324
            2526272829301
            2345678


            專注即時通訊及網游服務端編程
            ------------------------------------
            Openresty 官方模塊
            Openresty 標準模塊(Opm)
            Openresty 三方模塊
            ------------------------------------
            本博收藏大部分文章為轉載,并在文章開頭給出了原文出處,如有再轉,敬請保留相關信息,這是大家對原創作者勞動成果的自覺尊重??!如為您帶來不便,請于本博下留言,謝謝配合。

            常用鏈接

            留言簿(1)

            隨筆分類

            隨筆檔案

            相冊

            Awesome

            Blog

            Book

            GitHub

            Link

            搜索

            •  

            積分與排名

            • 積分 - 219199
            • 排名 - 117

            最新評論

            閱讀排行榜

            來自:http://blog.csdn.net/andylau00j/article/details/53769446
            Golang1.8官方支持加載動態庫了,而且看起來功能很強大。 文檔上面描述Plugin功能是協程并發安全的,而且支持高級數據類型(包括chan),同時不需要寫任何C代碼(以前或多或少需要寫一些)。


            下面我們通過一些demo來看看如何使用golang plugin功能
            入門
            下面是一個簡單的plugin 代碼:
            package main
            // // No C code needed.
            import "C"
            import (
                "fmt"
            )
            var V int
            func F() { fmt.Printf("Hello, number %d\n", V) }
            這是官方提供的例子,以后我們再慢慢擴展。
            go build -buildmode=plugin  main.go  //main.go是plugin文件名稱,你可以改成其它名
            你會看到多出一個main.so動態庫。
            然后我們開始調用這個動態庫
            package main
            import (
                "plugin"
            )
            func main() {
                p, err := plugin.Open("main.so")
                if err != nil {
                    panic(err)
                }
                v, err := p.Lookup("V")
                if err != nil {
                    panic(err)
                }
                f, err := p.Lookup("F")
                if err != nil {
                    panic(err)
                }
                *v.(*int) = 7
                f.(func())() 
            正常編譯這個go文件,然后執行它就可以看到屏幕會輸出
            Hello, number 7
            這只是最簡單的形式,下面我們增加一些難度,使用一些復雜數據類型。
            使用復雜類型
            我們自定義一個結構體
            package data
            type VS struct {
                Name   string
                Age    int
                School string
            }
            這里需要注意,必須要把結構體放到另外一個package中,否則build plugin時,會抱結構體找不到。
            然后我們開始修改plugin代碼
            package main
            // // No C code needed.
            import "C"
            import (
                "fmt"
                "temp/data"
            )
            var V int
            func F() { fmt.Printf("Hello, number %d\n", V) }
            var Vs data.VS
            func ComplexType() {
                fmt.Println(Vs.Name, Vs.Age, Vs.School)
            }
            我們聲明了VS結構體,然后增加了一個函數用來使用結構體(需要重新編譯plugin)。 調用方也需要同步修改
            package main
            import (
                "plugin"
                "temp/data"
            )
            func main() {
                p, err := plugin.Open("main.so")
                if err != nil {
                    panic(err)
                }
                v, err := p.Lookup("V")
                if err != nil {
                    panic(err)
                }
                f, err := p.Lookup("F")
                if err != nil {
                    panic(err)
                }
                *v.(*int) = 7
                f.(func())() // prints "Hello, number 7"
                vs, err := p.Lookup("Vs")
                if err != nil {
                    panic(err)
                }
                ct, err := p.Lookup("ComplexType")
                if err != nil {
                    panic(err)
                }
                *vs.(*data.VS) = data.VS{
                    Name:   "DATA",
                    Age:    11,
                    School: "BEIDA",
                }
                ct.(func())()
            }
            在main中我們構建了一個VS結構體,然后給各個屬性賦值。 如果正常,你將看到如下的信息
            Hello, number 7
            DATA 11 BEIDA
            到此為止,我們已經可以在Plugin中使用高級復雜結構體了,可以說已經滿足50%需求了。 但探索無止境,如果調用和Plugin之間需要互相通訊怎么辦?下面我們看看Plugin是否支持Chan。
            支持Chan
            首先聲明一個全局Chan
            package data
            type VS struct {
                Name   string
                Age    int
                School string
            }
            var Msg = make(chan VS)
            不但聲明了一個全局chan,而且是高級類型的chan。 下面在Plugin中使用chan
            package main
            // // No C code needed.
            import "C"
            import (
                "fmt"
                "temp/data"
            )
            var V int
            func F() { fmt.Printf("Hello, number %d\n", V) }
            var Vs data.VS
            func ComplexType() {
                fmt.Println(Vs.Name, Vs.Age, Vs.School)
                Vs.Age = Vs.Age * 4
                data.Msg <- Vs
            }
            Plugin接收到vs數據后,將AGE放大4倍在扔出去(扔哪里就不管了)(需要重新編譯plugin)。 再修改一下調用函數:
            package main
            import (
                "plugin"
                "temp/data"
            )
            func main() {
                p, err := plugin.Open("main.so")
                if err != nil {
                    panic(err)
                }
                v, err := p.Lookup("V")
                if err != nil {
                    panic(err)
                }
                f, err := p.Lookup("F")
                if err != nil {
                    panic(err)
                }
                *v.(*int) = 7
                f.(func())() // prints "Hello, number 7"
                vs, err := p.Lookup("Vs")
                if err != nil {
                    panic(err)
                }
                ct, err := p.Lookup("ComplexType")
                if err != nil {
                    panic(err)
                }
                *vs.(*data.VS) = data.VS{
                    Name:   "DATA",
                    Age:    11,
                    School: "BEIDA",
                }
                go ct.(func())()
                select {
                case m := <-data.Msg:
                    println(m.Age)
                }
            }
            其實是在main函數中接受扔回來的VS結構體。 編譯之后再執行,可以看到
            Hello, number 7
            DATA 11 BEIDA
            44
            OK,執行正常,符合預期。 這里先提供plugin如何使用,以后我們再剖析一下Golang是如何實現這些功能的。

            官方文檔:golang插件機制(plugin)
            https://tip.golang.org/pkg/plugin/?utm_source=tuicool&utm_medium=referral
            posted on 2017-01-19 18:44 思月行云 閱讀(2418) 評論(0)  編輯 收藏 引用 所屬分類: Golang
            99久久人妻无码精品系列蜜桃| 精品久久久无码人妻中文字幕豆芽| 婷婷久久综合九色综合98| 久久国产亚洲精品麻豆| 亚洲另类欧美综合久久图片区| 无码任你躁久久久久久老妇App| 久久亚洲精精品中文字幕| 久久中文字幕一区二区| 久久人妻AV中文字幕| 国产精品久久久99| 久久亚洲中文字幕精品有坂深雪| 久久精品一区二区三区中文字幕| 欧美精品乱码99久久蜜桃| 91精品国产91久久久久久蜜臀| 久久婷婷色综合一区二区| 国产激情久久久久影院小草| 蜜臀av性久久久久蜜臀aⅴ麻豆 | 精品国产VA久久久久久久冰 | 久久久久人妻精品一区三寸蜜桃| 国产精品99久久久精品无码| 久久精品无码一区二区日韩AV| 久久精品国产99久久久| 伊人久久精品无码av一区| 久久精品国产亚洲αv忘忧草| 国产国产成人久久精品| 久久免费精品视频| 久久精品国产久精国产| MM131亚洲国产美女久久| 综合人妻久久一区二区精品| 国产精品久久久久a影院| 久久久精品国产Sm最大网站| 91精品国产高清久久久久久国产嫩草 | 亚洲精品美女久久久久99小说| 日本福利片国产午夜久久| 国产精品久久久久久福利漫画| 久久夜色精品国产噜噜噜亚洲AV| 无码人妻精品一区二区三区久久久| 无码精品久久一区二区三区| 久久精品视频一| 亚洲人成精品久久久久| 91精品国产色综合久久|