登录
首页 >  Golang >  Go问答

处理网络数据包读取超时后的错误

来源:stackoverflow

时间:2024-04-30 13:39:39 497浏览 收藏

“纵有疾风来,人生不言弃”,这句话送给正在学习Golang的朋友们,也希望在阅读本文《处理网络数据包读取超时后的错误》后,能够真的帮助到大家。我也会在后续的文章中,陆续更新Golang相关的技术文章,有好的建议欢迎大家在评论留言,非常感谢!

问题内容

我正在使用conn.setreaddeadline方法来设置conn的读取超时,当conn.read等待超过指定时间时,就会返回并返回*net.operror类型的错误。此错误是 net 包在包装所有非 io.eof 错误后返回的。

我可以在使用 unwrap() 包装之前得到错误。超时错误是 *poll.deadlineexceedederror 类型的错误。我在代码中使用这样的语句来精确处理超时错误。

import "internal/poll"

_, err = conn.Read(p)

    if err != nil {
        if pe, ok := err.(*net.OpError); ok {
            err = pe.Unwrap()
            if timeout, ok := err.(*poll.DeadlineExceededError); ok {
                log.Error(fmt.Sprintf("%T, %s", timeout, timeout))
            }
        }
        return
    }

运行程序时,我收到了 use of inside package inside/poll not allowed 错误。编译器告诉我不能使用内部包。

我用google搜索找到了删除internal文件夹的解决方案,这是最终的解决方案吗?有更好的解决方案吗?


正确答案


os 包将该错误导出为 os.ErrDeadlineExceeded(检查 source code)。您可以尝试:

if errors.is(err, os.errdeadlineexceeded) {
    log.error("timeout error")
}

[编辑]实际上,在阅读@brit的评论后,这是检查该错误的记录方法。请参阅 Conn.SetDeadline() 的文档:

如果超过截止日期,则调用 read 或 write 或其他 i/o 方法将返回一个包含 os.errdeadlineexceeded 的错误。 这可以使用 error.is(err, os.errdeadlineexceeded) 进行测试。

另一个表明错误是“超时”错误的迹象是它是否有 timeout() bool 方法,该方法在调用时返回 true

这是 net.Error 接口的一部分(尽管该接口有一个额外的方法,已被记录为已弃用),并由 net.OpError 类型(以及内部 poll.DeadlineExceededError 类型)实现。

有几个函数(例如 net.OpError.IsTimeout()os.SyscallError.IsTimeout() 和公共函数 os.IsTimeout())可以通过直接将错误值转换为 timeout 接口来实现超时检查。

如果您想处理可以相互包装的错误,您可能需要使用 errors.as(...) 实现自己的 istimeout() 检查:

// isTimeout : use errors.As() to unwrap errors and check if a sub error is a Timeout error
func isTimeout(err error) bool {
    var terr interface{ Timeout() bool }
    return errors.As(err, &terr) && terr.Timeout()
}

https://go.dev/play/p/OhhKY3XsGjZ

请注意:

上面的 istimeout() 函数和 errors.is(err, os.errdeadlineexceeded) 调用之间的区别在于后者会尝试精确匹配 errdeadlineexceeded (这主要是通过设置 setdeadline() 触发的)上一些类似文件或连接的对象),而后一个可能会因为其他原因尝试宣传“我是超时错误”的错误返回 true(例如:http“408 请求超时”响应)

注意:以上所有链接均指 go 1.18.3 。根据您使用的 go 版本,您可能需要调整上面的一些代码(例如:在 go1.15 中添加了 os.errdeadlineexceeded

终于介绍完啦!小伙伴们,这篇关于《处理网络数据包读取超时后的错误》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>