Go网络开发中的两个技术点

鸟窝 2018-03-08 15:59

最近碰到群里网友问如果检查网络连接的Timeout,联想最近看到的另一个问题, 在这篇文章统一记录一下。

如果检查一个网络错误是Timeout导致的?

自 Go 1.6开始, 所有的超时导致的网络错误都可以通过net.Error的Timeout()方法检查。

123
if err, ok := err.(net.Error); ok && err.Timeout() {    ……}

更早版本的Go并没有专门的检查Timeout方法。

可以参考:https://stackoverflow.com/questions/23494950/specifically-check-for-timeout-error

产品中一定不要使用默认的http.Get

如果你觉得方便,直接使用http.Get或者类似的方法发送请求,可能会导致一些问题, 因为这默认是使用DefaultClient作为client:

  1. 多goroutine共享,这意味着在别处对DefaultClient的改动会影响你当前的使用
  2. 未设置connection timeout和 read/write timeout
  3. 默认的idle connection等设置可能不满足你的需求

一个具体的使用默认的client导致的问题可以参考Don’t use Go’s default HTTP client (in production)

所以Go开发过程中最好使用自己定制的Client:

1234567891011121314151617181920212223
var timeout = time.Duration(2 * time.Second)func dialTimeout(network, addr string) (net.Conn, error) {    return net.DialTimeout(network, addr, timeout)}func main() {    transport := http.Transport{        Dial: dialTimeout,        Proxy: ...,        MaxIdleConns: ...,        MaxIdleConnsPerHost: ...,        IdleConnTimeout: ...,        ResponseHeaderTimeout: ...,        DisableCompression:...,    }    client := http.Client{        Transport: &transport,    }    resp, err := client.Get("http://some.url")}

酌情设置Transport的参数。

[返回] [原文链接]