本文主要是介绍正确处理Go语言中的瞬时资源,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
开发人员在代码中经常要和那些需要在某些时刻进行关闭的瞬时(临时)资源打交道。例如,为了避免出现磁盘或内存泄漏,结构体一般可以实现io.Close接口来表达一个瞬时资源必须被关闭。让我们看几个常见的例子,看看当资源没有被正确关闭会发生什么,以及该如何正确处理它们。
1. HTTP Body
首先,让我们在HTTP的上下文中讨论这个问题。我们将编写一个getBody方法,发出HTTP GET请求并返回HTTP正文响应。下面是第一个实现:
type handler struct {client http.Clienturl string
}func (h handler) getBody() (string, error) {resp, err := h.client.Get(h.url)if err != nil {return "", err}body, err := io.ReadAll(resp.Body)if err != nil {return "", err}return string(body), nil
}
我们使用http.Get并通过 io.ReadAll 解析响应。这个方法看起来不错,而且它正确地返回了HTTP响应体。然而,这里出现了资源泄漏。让我们来了解一下哪里出了问题。
resp 是*http.Response 类型的。它包含一个Body io.ReadCloser 字段(io.ReadCloser同时实现了 io.Reader 和 io.Closer)。如果http.Get没有返回错误,Body 变量必须被关闭,否则,就会出现资源泄漏。在这种情况下,我们的应用程序将保留一些被分配的内存,即使这些内存不再需要,也不会被GC回收。在最坏的情况下,可能会阻止客户端重复使用TCP连接。
处理响应体关闭最方便的方式是使用defer语句:
defer func() {err := resp.Body.Close()if err != nil {log.Printf("failed to close response: %v\n", err)}
}()
在这个实现中,我们将响应体资源的闭合处理为一个defer函数,一旦getBody返回它就会被执行。
注意:
在服务端,当实现一个HTTP处理程序时,我们不需要关闭请求体,因为服务端会自动将其关闭。
我们还
这篇关于正确处理Go语言中的瞬时资源的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!