为什么要使用Deadlines

当我们使用gRPC时,gRPC库关系的是连接,序列化,反序列化和超时执行。Deadlines 允许gRPC客户端设置自己等待多长时间来完成rpc操作,直到出现这个错误 DEADLINE_EXCEEDED。但是在正常情况下,这个DEADLINE_EXCEEDED默认设置是一个很大的数值。

一些语言的API用deadline,一些用 timeout。

在正常情况下,你没有设置deadline,那么所有的请求可能在最大请求时间过后才超时。这样你对于你的服务器资源,可能存在风险,比如内存,可能因为这个长期运行的服务而增长很快,从而耗尽资源。

为了避免这种情况,需要给你的客户端请求程序设置一个默认的超时时间,在这段时间内请求没有返回,那么就超时报错。

怎么使用?Deadlines使用步骤

使用步骤:

1.设置deadlines

var deadlineMs = flag.Int("deadline_ms", 20*1000, "Default deadline in milliseconds.")

clientDeadline := time.Now().Add(time.Duration(*deadlineMs) * time.Millisecond)
ctx, cancel := context.WithDeadline(ctx, clientDeadline)

2.检查deadlines

if ctx.Err() == context.Canceled {
return status.New(codes.Canceled, "Client cancelled, abandoning.")
}

具体使用:

1. 建立连接时超时控制:

客户端建立连接时,使用的Dial()函数,它位于

google.golang.org/grpc/clientconn.go 中,我们看看这个函数内容:

func Dial(target string, opts ...DialOption) (*ClientConn, error) {    
return DialContext(context.Background(), target, opts...)
}

它里面调用的 DialContext() 函数,这个函数非常长,他们在同一个文件中,它是实际执行的函数,这里面就有context的timeout和Done相关操作。你也可以到google.golang.org/grpc/clientconn.go文件中去看看这个函数DialContext具体是干嘛的。

使用的时候传入设置timeout的context,如下:

ctx, cancel := context.Timeout(context.Bakcground(), time.Second*5)
defer cancel()
conn, err := grpc.DialContext(ctx, address, grpc.WithBlock(), grpc.WithInsecure())
  • grpc.WithInsecure() ,这个参数啥意思?

    gRPC是建立在HTTP/2上的,所以对TLS提供了很好的支持。如果在客户端建立连接过程中设置 grpc.WithInsecure() 就可以跳过对服务器证书的验证。写练习时可以用这个参数,但是在真实的环境中,不要这样做,因为有泄露信息的风险。
  • grpc.WithBlock()

    这个参数会阻塞等待握手成功。

    因为用Dial连接时是异步连接,连接状态为正在连接,如果设置了这个参数就是同步连接,会阻塞等待握手成功。

    这个还和超时设置有关,如果你没有设置这个参数,那么context超时控制将会失效。

2. 调用时超时:

函数的调用超时控制

ctx, cancel := context.WithTimeout(context.TODO(), time.Second*5)
defer cancel()
result, err := c.SayHello(ctx, &pb.HelloRequest{Name: name})

实例

用grpc官方的例子来练习下

目录结构:

grpc-tutorial
-- 04deadlines
--client
- main.go
--server
- main.go
--proto/echo
- echo.proto
- echo.pb.go

1.首先是定义服务 echo.proto

syntax = "proto3";
package echo;message EchoRequest {   
string message = 1;
} message EchoResponse {   
string message = 1;
} service Echo {   
rpc UnaryEcho(EchoRequest) returns (EchoRequest) {}   
rpc ServerStreamingEcho(EchoRequest) returns (stream EchoResponse) {}   
rpc ClientStreamingEcho(stream EchoRequest) returns (EchoResponse) {}   
rpc BidirectionalStreamingEcho(stream EchoRequest) returns (stream EchoResponse){}
}

进入到proto/echo目录,生成go文件,命令如下:

protoc -I . --go_out=plugins=grpc:. ./echo.proto

2.客户端

client\main.go

有2个主要的函数,2端都是stream和都不是stream,先看都不是stream的函数

都不是stream的函数

// unaryCall 不是stream的请求
func unaryCall(c pb.EchoClient, requestID int, message string, want codes.Code) {   
ctx, cancel := context.WithTimeout(context.Background(), time.Second) //超时设置   
defer cancel()    req := &pb.EchoRequest{Message: message} //参数部分 _, err := c.UnaryEcho(ctx, req) //调用函数发送请求给服务端
got := status.Code(err)   //
fmt.Printf("[%v] wanted = %v, got = %v\n", requestID, want, got)
}

上面的code设置在文件 grpc/codes/codes.go

type Code uint32

const (
OK Code = 0
Canceled Code = 1
Unknown Code = 2
InvalidArgument Code = 3
DeadlineExceeded Code = 4
... ...
)

2端都是stream的函数:

// streamingCall,2端都是stream
func streamingCall(c pb.EchoClient, requestID int, message string, want codes.Code) {
ctx, cancel := context.WithTimeout(context.Background(), time.Second)//超时设置
defer cancel() stream, err := c.BidirectionalStreamingEcho(ctx)//双向stream
if err != nil {
log.Printf("Send error : %v", err)
return
} err = stream.Send(&pb.EchoRequest{Message: message})//发送
if err != nil {
log.Printf("Send error : %v", err)
return
} _, err = stream.Recv() //接收 got := status.Code(err)
fmt.Printf("[%v] wanted = %v, got = %v\n", requestID, want, got)
}

main 执行函数

func main() {
flag.Parse() conn, err := grpc.Dial(*addr, grpc.WithInsecure(), grpc.WithBlock())
if err != nil {
log.Fatalf("did not connect : %v ", err)
}
defer conn.Close() c :=pb.NewEchoClient(conn) // 成功请求
unaryCall(c, 1, "word", codes.OK)
// 超时 deadline
unaryCall(c, 2, "delay", codes.DeadlineExceeded)
// A successful request with propagated deadline
unaryCall(c, 3, "[propagate me]world", codes.OK)
// Exceeds propagated deadline
unaryCall(c, 4, "[propagate me][propagate me]world", codes.DeadlineExceeded)
// Receives a response from the stream successfully.
streamingCall(c, 5, "[propagate me]world", codes.OK)
// Exceeds propagated deadline before receiving a response
streamingCall(c, 6, "[propagate me][propagate me]world", codes.DeadlineExceeded)
}

3.服务端

定义一个struct

type server struct {   
pb.UnimplementedEchoServer   
client pb.EchoClient   
cc *grpc.ClientConn
}

2端不是stream的函数:

func (s *server) UnaryEcho(ctx context.Context, req *pb.EchoRequest)(*pb.EchoResponse, error) {
message := req.Message
if strings.HasPrefix(message, "[propagate me]") {//判断接收的值
time.Sleep(800 * time.Millisecond)
message := strings.TrimPrefix(message, "[propagate me]")
return s.client.UnaryEcho(ctx, &pb.EchoRequest{Message:message}) //<1>
} if message == "delay" {
time.Sleep(1500 * time.Millisecond) // message=delay 时睡眠1500毫秒,大于client的设置的1秒,这里就超时了
} return &pb.EchoResponse{Message:message}, nil
}

上面函数标注 <1> 这个地方比较有意思,当client端发送的字符串包含 [propagate me] 字符串时,先睡眠800毫秒,然后在重新执行客户端请求服务端的函数 s.client.UnaryEcho() , 在次运行到服务端的 UnaryEcho(),客户端已经超时了。

也就是说client/main.go 先请求了一次服务端,然后在server/main.go 的函数 func (s *server) UnaryEcho(ctx context.Context, req *pb.EchoRequest) 又执行了一次请求服务端,所以会导致超时。

2端设置stream的函数:

func (s *server) BidirectionalStreamingEcho(stream pb.Echo_BidirectionalStreamingEchoServer) error {
for {
req, err := stream.Recv()
if err == io.EOF {
return status.Error(codes.InvalidArgument, "request message not received")
}
if err != nil {
return err
} message := req.Message
if strings.HasPrefix(message, "[propagate me]") {
time.Sleep(800 * time.Millisecond)
message = strings.TrimPrefix(message, "[propagate me]")
res, err := s.client.UnaryEcho(stream.Context(), &pb.EchoRequest{Message:message})//再次执行客户端请求服务端函数,这里可能会超时
if err != nil {
return err
}
stream.Send(res)
} if message == "delay" {
time.Sleep(1500 * time.Millisecond)
}
stream.Send(&pb.EchoResponse{Message:message})
}
}

main函数

func main() {
flag.Parse() address := fmt.Sprintf(":%v", *port)
lis, err := net.Listen("tcp", address)
if err != nil {
log.Fatalf("failed to listen: %v ", err)
} echoServer := newEchoServer()
defer echoServer.Close() grpcServer := grpc.NewServer()
pb.RegisterEchoServer(grpcServer, echoServer) if err := grpcServer.Serve(lis); err != nil {
log.Fatalf("failed to serve: %v ", err)
}
}

执行

先运行 /server/main.go , go run main.go

在运行 /client/main.go, go run main.go

执行结果:

 go run main.go
[1] wanted = OK, got = OK
[2] wanted = DeadlineExceeded, got = DeadlineExceeded
[3] wanted = OK, got = Unavailable
[4] wanted = DeadlineExceeded, got = Unavailable
[5] wanted = OK, got = Unavailable
[6] wanted = DeadlineExceeded, got = Unavailable

gRPC 系列代码地址:

参考:

最新文章

  1. JS:事件对象1
  2. python 反射器
  3. 从分析SQLSERVER ERRORLOG查找错误折射出的工作效率问题
  4. POJ 1182 食物链 (三态种类并查集)
  5. 初学structs2,结果类型简单示例
  6. IIS7 / IIS7.5 URL 重写 HTTP 重定向到 HTTPS(转)
  7. Contest2037 - CSU Monthly 2013 Oct (problem F :ZZY and his little friends)
  8. iOS设计模式解析(六)代理模式
  9. 解决 Mybatis 元素类型为 &quot;resultMap&quot; 的内容必须匹配 &quot;(constructor?,id*,result*,association*,collection*,discriminat
  10. RF无线射频电路设计干货分享
  11. redis 设置分布式锁要避免死锁
  12. 求一个Map中最大的value值,同时列出键,值
  13. Yii2基本概念之——行为(Behavior)
  14. css 如何让背景图片拉伸填充避免重复显示
  15. 批量配置SSH互信脚本
  16. linux中,当执行rpm -e删除一个软件包时,都做了些什么事
  17. js监控鼠标滚动事件
  18. JS频率控制函数
  19. 找到SVN版本机上项目的地址
  20. Jupyter Notebook的使用

热门文章

  1. Jexl表达式引擎-根据字符串动态执行JAVA.md
  2. vs coed的使用(二) 如何运行cpp文件(不用插件比如code runner)
  3. git的分支远程连接和远程分支的拉取推送及冲突处理
  4. MySQL之字段数据类型和列属性
  5. 题解 P1201 【[USACO1.1]贪婪的送礼者Greedy Gift Givers】
  6. lua中单引号和双引号和/的输出的问题
  7. 在Dockerfile中使用和“Source”的Run指令不起作用?
  8. Ribbon 负载规则替换
  9. PDOStatement::rowCount
  10. 牛客练习赛63 牛牛的斐波那契字符串 矩阵乘法 KMP