Go实现超时控制

应用场景

交易、金融等事务系统往往会有各种下游,绝大多数时候我们会以同步方式进行访问,如调用RPC、HTTP等。 这些下游在通常延时相对稳定,但有时可能出现极端的超大延时,这些极端case可能具备特定的业务特征,也有可能单纯是硬件、网络的问题造成,最终表现在系统P99或者P999的延时出现了突刺,如果是面向C端的场景,也会向用户报出一些系统错误,造成用户体验的下降。 一种简易的解决方案是,针对关键的下游节点增加超时控制。在特定时间内,如果下游到期还未返回,不再暴露系统级错误,而是做特殊化处理,比如返回「处理中」状态。

Go实现方案

设计一个方法,使用闭包,传入时间和执行的任务,如果任务执行完未到时间,则直接返回,否则通知调用者超时。 为了保证代码简介和使用简单,我们仅定义一个Wrapper方法,方法定义如下 func TimeoutControlWrapper(duration time.Duration, fn func()) (timeout bool)

官方包time有一个After方法,可以在指定时间内,返回一个channel,基于此来判断是否超时。 另外,在Wrapper方法里异步化执行目标方法,执行完成后写入一个finish信号通知。 同时监听这两个channel,判断是否超时,代码如下

go 复制代码
func TimeoutControlWrapper(duration time.Duration, fn func()) (timeout bool) {
        finish := make(chan struct{})
        go func() {
                fn()
                finish <- struct{}{}
        }()

        select {
        case <-finish:
                return false
        case <-time.After(duration):
                return true
        }
}

结合场景,假设系统会调用一个支付系统的接口,接口本身延时不稳定,因此我们套用TimeoutControlWrapper

go 复制代码
func CallPaymentSystem(param PayParam) (payStatus PayStatus) {
    var payStatus PayStatus
    timeout := TimeoutControlWrapper(time.Second, func() {
        payStatus = PaymentSystemRPC.Pay(param)
    })
    if timeout {
        warn()    // WARN告警
        return  PROCESSING  // 返回处理中
    }
    return payStatus
}

延伸思考

上述通过一个简单的Wrapper,来实现调用下游时的超时控制。但在引入的场景里,实现上是不严谨的。哪怕不增加超时控制,我们也无法确认请求是否真实到达了下游系统,这本质上是一个分布式事务的问题,需要我们设计更加健全的系统能力保证一致性,比如通过消息的方式、补偿机制、增加对账系统。

相关推荐
gopyer16 小时前
180课时吃透Go语言游戏后端开发7:Go语言中的函数
开发语言·游戏·golang·go·函数
lypzcgf1 天前
Coze源码分析-资源库-编辑工作流-后端源码-流程/技术/总结
go·源码分析·工作流·coze·coze源码分析·ai应用平台·agent平台
Penge6663 天前
Go语言中的切片展开操作符 ...
后端·go
野木香3 天前
go语言websocket连接,重连,发心跳示例
websocket·go
豆浆whisky4 天前
netpoll性能调优:Go网络编程的隐藏利器|Go语言进阶(8)
开发语言·网络·后端·golang·go
豆浆Whisky4 天前
Go实现百万级连接:资源管控与性能平衡的艺术|Go语言进阶(9)
后端·go
CAir25 天前
CGO 原理
c++·go·cgo
李广坤6 天前
Go语言学习第二课:数据类型和变量
go
lypzcgf6 天前
Coze源码分析-资源库-删除数据库-后端源码-领域服务/数据访问层
数据库·go·coze·coze源码分析·智能体平台·ai应用平台·agent平台
qinyuan157 天前
生产环境go-redsync使用示例
后端·go