Go Micro(3)——开发微服务

Go Micro(3)——开发微服务

这是一个高等级的说明:怎样使用 go-micro 来编写微服务,如果你想学习更多微服务的知识以及Micro的整体架构,参考以前的文章。

什么是 Go Micro?

Go Micro 是一个插件化的基础框架,基于此可以构建微服务。Micro 的设计哲学是『可插拔』的插件化架构。在架构之外,它默认实现了 consul 作为服务发现,通过 http 进行通信,通过 protobufjson 进行编解码。我们一步步深入下去。

Go Micro 是:

  • 一个用 Golang 编写的包
  • 一系列插件化的接口定义
  • 基于 RPC

Go Micro 为下面的模块定义了接口:

  • 服务发现
  • 编解码
  • 服务端、客户端
  • 订阅、发布消息

更详细的说明可以在这里看到。

Go Micro 从一年多以前开始开发,最初只是个人需求,很快我发现这对那些编写微服务的程序员会有很大的价值。它基于我在不同的技术公司如 googlehailo 的开发经验编写而成。

就像前面提到的,Go Micro 是一个 golang 编写的插件化架构,专注于提供底层的接口定义和基础工具。这些接口可以接纳各种实现。比如 Registry 接口定义了服务发现的接口,默认采用了 consul 作为服务发现的实现,但也可以采用其他实现比如 etcdzookeeper 等,只要能满足接口,就可以使用。

插件化的架构意味着如果你想替换底层的实现,你不需要修改任何底层的代码。

编写一个服务

如果你想直接看代码,看这里:examples/service

顶层的 Service 接口是构建服务的主要组件。它把底层的各个包需要实现的接口,做了一次封装。

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

初始化

一个服务可以这样创建 micro.NewService

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

service := micro.NewService()

参数可以在创建时传入

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

所有可选的参数设置可以在这里看到

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 来解析参数,附加的处理可以通过 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 提供了提供了预定义的参数,也会被 service.Init 解析,这里可以看到所有的 flag

定义 API

我们使用 protobuf 文件来定义服务的 API,这是一种方便且严格的定义方式,协议将会提供给服务端和客户端。下面是一个协议的例子:greeter.proto

syntax = "proto3";

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

message HelloRequest {
    string name = 1;
}

message HelloResponse {
    string greeting = 2;
}

这里定义了一个服务叫做 Greeter,它提供一个接口叫 Hello,它接受 HelloRequest 的请求,返回 HelloResponse

生成 API 接口

我们使用 protocproto-gen-go 这两个工具来生成代码,Go Micro 也会生成客户端代码,减少工作量,这里需要使用我们 fork 并修改过的 github.com/micro/protobuf,与原始版本的区别是,fork 版本能生成客户端代码

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

生成的代码可以在 handler 中引用相应的包进行使用,下面是生成的一部分代码

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}))
}

实现 handler

服务端需要注册 handler 来处理请求,一个 handler 是一个这样的方法:

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

正如上面看到的,一个 handler 实现了 API 协议中定义的接口

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

这里是一个 Greeterhandler 实现

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 需要注册到某个服务

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

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

运行服务

服务可以直接调用 server.Run() 来运行,这会让服务监听一个随机端口,这个调用也会让服务将自身注册到注册器,当服务停止运行时,会在注册器注销自己。

完整的服务端

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)
        }
}

注意,服务发现机制需要首先运行起来,这样服务才能注册到注册器中,才能被客户端发现。

编写客户端

client 包用于向服务端发起请求,当你创建一个服务,客户端可以调用的接口已经自动生成了

调用上面的服务可以用下面的客户端代码

/ 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 就是我们刚才生成的代码,根据服务名称发送请求。

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容

  • 微服务工具箱 现在你也许听到了这个新现象:微服务。如果你对此不熟悉也有兴趣学习,欢迎参考上一篇文章。 这篇文章我们...
    浮x尘阅读 6,245评论 0 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,580评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,392评论 25 707
  • 他发一个吐血的表情,说没事...... 我亦有些莫名其妙,想问因果又觉得无此必要,就闲扯一般和他聊聊各自的现状。 ...
    小瓶盖shy阅读 267评论 1 1
  • 大一下, 无聊, 无语, 无心, 开心, 开车, 开会。 不, 会变的。
    Aolando阅读 140评论 0 0