猿问

在 Go 中检测连接断开的正确方法是什么?

我正在使用从移动客户端读取上传的 Go HTTP 服务器实现。但是,我遇到了一些问题,因为长时间保持活动状态,如果移动客户端脱机(经常发生),服务器将挂起读取请求缓冲区很长时间。

检测断开的连接并关闭输入缓冲区的正确方法是什么?


开心每一天1111
浏览 466回答 2
2回答

斯蒂芬大帝

在服务器上设置一个合理的超时时间,例如:srv := &http.Server{    Addr: ":443",    ReadTimeout: time.Minute * 2,    WriteTimeout: time.Minute * 2,}log.Fatal(srv.ListenAndServeTLS(certFile, keyFile))

慕容森

因为我只想在写入停止时断开连接(并且很快,以便我可以记录到目前为止收到的数据并允许客户端恢复),所以 ReadTimeout 对我来说不是正确的解决方案。我在这个要点中找到了答案。您需要在连接本身上设置读取。包网络超时import (    "net"    "time")// Listener wraps a net.Listener, and gives a place to store the timeout// parameters. On Accept, it will wrap the net.Conn with our own Conn for us.type Listener struct {    net.Listener    ReadTimeout  time.Duration    WriteTimeout time.Duration}func (l *Listener) Accept() (net.Conn, error) {    c, err := l.Listener.Accept()    if err != nil {        return nil, err    }    tc := &Conn{        Conn:         c,        ReadTimeout:  l.ReadTimeout,        WriteTimeout: l.WriteTimeout,    }    return tc, nil}// Conn wraps a net.Conn, and sets a deadline for every read// and write operation.type Conn struct {    net.Conn    ReadTimeout  time.Duration    WriteTimeout time.Duration}func (c *Conn) Read(b []byte) (int, error) {    err := c.Conn.SetReadDeadline(time.Now().Add(c.ReadTimeout))    if err != nil {        return 0, err    }    return c.Conn.Read(b)}func (c *Conn) Write(b []byte) (int, error) {    err := c.Conn.SetWriteDeadline(time.Now().Add(c.WriteTimeout))    if err != nil {        return 0, err    }    return c.Conn.Write(b)}func NewListener(addr string, readTimeout, writeTimeout time.Duration) (net.Listener, error) {    l, err := net.Listen("tcp", addr)    if err != nil {        return nil, err    }    tl := &Listener{        Listener:     l,        ReadTimeout:  readTimeout,        WriteTimeout: writeTimeout,    }    return tl, nil}
随时随地看视频慕课网APP

相关分类

Go
我要回答