Go Micro(3)——開發(fā)微服務(wù)

Go Micro(3)——開發(fā)微服務(wù)

這是一個高等級的說明:怎樣使用 go-micro 來編寫微服務(wù)箱玷,如果你想學(xué)習(xí)更多微服務(wù)的知識以及Micro的整體架構(gòu),參考以前的文章汰蓉。

什么是 Go Micro?

Go Micro 是一個插件化的基礎(chǔ)框架喜鼓,基于此可以構(gòu)建微服務(wù)。Micro 的設(shè)計哲學(xué)是『可插拔』的插件化架構(gòu)溜歪。在架構(gòu)之外,它默認(rèn)實現(xiàn)了 consul 作為服務(wù)發(fā)現(xiàn)许蓖,通過 http 進行通信蝴猪,通過 protobufjson 進行編解碼。我們一步步深入下去膊爪。

Go Micro 是:

  • 一個用 Golang 編寫的包
  • 一系列插件化的接口定義
  • 基于 RPC

Go Micro 為下面的模塊定義了接口:

  • 服務(wù)發(fā)現(xiàn)
  • 編解碼
  • 服務(wù)端自阱、客戶端
  • 訂閱、發(fā)布消息

更詳細(xì)的說明可以在這里看到米酬。

Go Micro 從一年多以前開始開發(fā)沛豌,最初只是個人需求,很快我發(fā)現(xiàn)這對那些編寫微服務(wù)的程序員會有很大的價值赃额。它基于我在不同的技術(shù)公司如 googlehailo 的開發(fā)經(jīng)驗編寫而成加派。

就像前面提到的,Go Micro 是一個 golang 編寫的插件化架構(gòu)爬早,專注于提供底層的接口定義和基礎(chǔ)工具哼丈。這些接口可以接納各種實現(xiàn)启妹。比如 Registry 接口定義了服務(wù)發(fā)現(xiàn)的接口筛严,默認(rèn)采用了 consul 作為服務(wù)發(fā)現(xiàn)的實現(xiàn),但也可以采用其他實現(xiàn)比如 etcdzookeeper 等饶米,只要能滿足接口桨啃,就可以使用。

插件化的架構(gòu)意味著如果你想替換底層的實現(xiàn)檬输,你不需要修改任何底層的代碼照瘾。

編寫一個服務(wù)

如果你想直接看代碼,看這里:examples/service

頂層的 Service 接口是構(gòu)建服務(wù)的主要組件丧慈。它把底層的各個包需要實現(xiàn)的接口析命,做了一次封裝。

type Service interface {
    Init(...Option)
    Options() Options
    Client() client.Client
    Server() server.Server
    Run() error
    String() string
}

初始化

一個服務(wù)可以這樣創(chuàng)建 micro.NewService

import "github.com/micro/go-micro"

service := micro.NewService()

參數(shù)可以在創(chuàng)建時傳入

service := micro.NewService(
    micro.Name("greeter"),
    micro.Version("latest"),
)

所有可選的參數(shù)設(shè)置可以在這里看到

Go Micro 也提供了讀取命令行的方式

import (
    "github.com/micro/cli"
    "github.com/micro/go-micro"
)

service := micro.NewService(
    micro.Flags(
        cli.StringFlag{
            Name:  "environment",
            Usage: "The environment",
        },
    )
)

通過 service.Init 來解析參數(shù)逃默,附加的處理可以通過 micro.Action 解決

service.Init(
    micro.Action(func(c *cli.Context) {
        env := c.StringFlag("environment")
        if len(env) > 0 {
            fmt.Println("Environment set to", env)
        }
    }),
)

Go Micro 提供了提供了預(yù)定義的參數(shù)鹃愤,也會被 service.Init 解析,這里可以看到所有的 flag

定義 API

我們使用 protobuf 文件來定義服務(wù)的 API完域,這是一種方便且嚴(yán)格的定義方式软吐,協(xié)議將會提供給服務(wù)端和客戶端。下面是一個協(xié)議的例子:greeter.proto

syntax = "proto3";

service Greeter {
    rpc Hello(HelloRequest) returns (HelloResponse) {}
}

message HelloRequest {
    string name = 1;
}

message HelloResponse {
    string greeting = 2;
}

這里定義了一個服務(wù)叫做 Greeter吟税,它提供一個接口叫 Hello凹耙,它接受 HelloRequest 的請求姿现,返回 HelloResponse

生成 API 接口

我們使用 protocproto-gen-go 這兩個工具來生成代碼肖抱,Go Micro 也會生成客戶端代碼备典,減少工作量,這里需要使用我們 fork 并修改過的 github.com/micro/protobuf意述,與原始版本的區(qū)別是熊经,fork 版本能生成客戶端代碼

go get github.com/micro/protobuf/{proto,protoc-gen-go}
protoc --go_out=plugins=micro:. greeter.proto

生成的代碼可以在 handler 中引用相應(yīng)的包進行使用,下面是生成的一部分代碼

type HelloRequest struct {
    Name string `protobuf:"bytes,1,opt,name=name" json:"name,omitempty"`
}

type HelloResponse struct {
    Greeting string `protobuf:"bytes,2,opt,name=greeting" json:"greeting,omitempty"`
}

// Client API for Greeter service

type GreeterClient interface {
    Hello(ctx context.Context, in *HelloRequest, opts ...client.CallOption) (*HelloResponse, error)
}


type greeterClient struct {
    c           client.Client
    serviceName string
}

func NewGreeterClient(serviceName string, c client.Client) GreeterClient {
    if c == nil {
        c = client.NewClient()
    }
    if len(serviceName) == 0 {
        serviceName = "greeter"
    }
    return &greeterClient{
        c:           c,
        serviceName: serviceName,
    }
}

func (c *greeterClient) Hello(ctx context.Context, in *HelloRequest, opts ...client.CallOption) (*HelloResponse, error) {
    req := c.c.NewRequest(c.serviceName, "Greeter.Hello", in)
    out := new(HelloResponse)
    err := c.c.Call(ctx, req, out, opts...)
    if err != nil {
        return nil, err
    }
    return out, nil
}

// Server API for Greeter service

type GreeterHandler interface {
    Hello(context.Context, *HelloRequest, *HelloResponse) error
}

func RegisterGreeterHandler(s server.Server, hdlr GreeterHandler) {
    s.Handle(s.NewHandler(&Greeter{hdlr}))
}

實現(xiàn) handler

服務(wù)端需要注冊 handler 來處理請求欲险,一個 handler 是一個這樣的方法:

func(ctx context.Context, req interface{}, rsp interface{}) error

正如上面看到的镐依,一個 handler 實現(xiàn)了 API 協(xié)議中定義的接口

type GreeterHandler interface {
        Hello(context.Context, *HelloRequest, *HelloResponse) error
}

這里是一個 Greeterhandler 實現(xiàn)

import proto "github.com/micro/micro/examples/service/proto"

type Greeter struct{}

func (g *Greeter) Hello(ctx context.Context, req *proto.HelloRequest, rsp *proto.HelloResponse) error {
    rsp.Greeting = "Hello " + req.Name
    return nil
}

handler 需要注冊到某個服務(wù)

service := micro.NewService(
    micro.Name("greeter"),
)

proto.RegisterGreeterHandler(service.Server(), new(Greeter))

運行服務(wù)

服務(wù)可以直接調(diào)用 server.Run() 來運行,這會讓服務(wù)監(jiān)聽一個隨機端口天试,這個調(diào)用也會讓服務(wù)將自身注冊到注冊器槐壳,當(dāng)服務(wù)停止運行時,會在注冊器注銷自己喜每。

完整的服務(wù)端

package main

import (
        "log"

        "github.com/micro/go-micro"
        proto "github.com/micro/go-micro/examples/service/proto"

        "golang.org/x/net/context"
)

type Greeter struct{}

func (g *Greeter) Hello(ctx context.Context, req *proto.HelloRequest, rsp *proto.HelloResponse) error {
        rsp.Greeting = "Hello " + req.Name
        return nil
}

func main() {
        service := micro.NewService(
                micro.Name("greeter"),
                micro.Version("latest"),
        )

        service.Init()

        proto.RegisterGreeterHandler(service.Server(), new(Greeter))

        if err := service.Run(); err != nil {
                log.Fatal(err)
        }
}

注意务唐,服務(wù)發(fā)現(xiàn)機制需要首先運行起來,這樣服務(wù)才能注冊到注冊器中带兜,才能被客戶端發(fā)現(xiàn)枫笛。

編寫客戶端

client 包用于向服務(wù)端發(fā)起請求,當(dāng)你創(chuàng)建一個服務(wù)刚照,客戶端可以調(diào)用的接口已經(jīng)自動生成了

調(diào)用上面的服務(wù)可以用下面的客戶端代碼

/ create the greeter client using the service name and client
greeter := proto.NewGreeterClient("greeter", service.Client())

// request the Hello method on the Greeter handler
rsp, err := greeter.Hello(context.TODO(), &proto.HelloRequest{
    Name: "John",
})
if err != nil {
    fmt.Println(err)
    return
}

fmt.Println(rsp.Greeter)

proto.NewGreeterClient 就是我們剛才生成的代碼刑巧,根據(jù)服務(wù)名稱發(fā)送請求。

完整的示例可以在這里看到:go-micro/examples/service

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末无畔,一起剝皮案震驚了整個濱河市啊楚,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌浑彰,老刑警劉巖恭理,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異郭变,居然都是意外死亡颜价,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進店門诉濒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來周伦,“玉大人,你說我怎么就攤上這事循诉『崃荆” “怎么了?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長狈蚤。 經(jīng)常有香客問我困肩,道長,這世上最難降的妖魔是什么脆侮? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任锌畸,我火速辦了婚禮,結(jié)果婚禮上靖避,老公的妹妹穿的比我還像新娘潭枣。我一直安慰自己,他們只是感情好幻捏,可當(dāng)我...
    茶點故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布盆犁。 她就那樣靜靜地躺著,像睡著了一般篡九。 火紅的嫁衣襯著肌膚如雪谐岁。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天榛臼,我揣著相機與錄音伊佃,去河邊找鬼。 笑死沛善,一個胖子當(dāng)著我的面吹牛航揉,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播金刁,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼帅涂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了胀葱?” 一聲冷哼從身側(cè)響起漠秋,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎抵屿,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體捅位,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡轧葛,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了艇搀。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片尿扯。...
    茶點故事閱讀 40,096評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖焰雕,靈堂內(nèi)的尸體忽然破棺而出衷笋,到底是詐尸還是另有隱情,我是刑警寧澤矩屁,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布辟宗,位于F島的核電站爵赵,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏泊脐。R本人自食惡果不足惜空幻,卻給世界環(huán)境...
    茶點故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望容客。 院中可真熱鬧秕铛,春花似錦、人聲如沸缩挑。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽供置。三九已至镜遣,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間士袄,已是汗流浹背悲关。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留娄柳,地道東北人寓辱。 一個月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓,卻偏偏與公主長得像赤拒,于是被迫代替她去往敵國和親秫筏。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,037評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 微服務(wù)工具箱 現(xiàn)在你也許聽到了這個新現(xiàn)象:微服務(wù)挎挖。如果你對此不熟悉也有興趣學(xué)習(xí)这敬,歡迎參考上一篇文章。 這篇文章我們...
    浮x塵閱讀 6,264評論 0 13
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理蕉朵,服務(wù)發(fā)現(xiàn)崔涂,斷路器,智...
    卡卡羅2017閱讀 134,659評論 18 139
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,167評論 25 707
  • 他發(fā)一個吐血的表情始衅,說沒事...... 我亦有些莫名其妙冷蚂,想問因果又覺得無此必要,就閑扯一般和他聊聊各自的現(xiàn)狀汛闸。 ...
    小瓶蓋shy閱讀 273評論 1 1
  • 大一下蝙茶, 無聊, 無語诸老, 無心隆夯, 開心, 開車, 開會蹄衷。 不忧额, 會變的。
    Aolando閱讀 140評論 0 0