当前位置: > > > > 处理网络数据包读取超时后的错误
来源:stackoverflow
2024-04-30 13:39:39
0浏览
收藏
“纵有疾风来,人生不言弃”,这句话送给正在学习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
包将该错误导出为 (检查 )。您可以尝试:
if errors.is(err, os.errdeadlineexceeded) { log.error("timeout error") }
[编辑]实际上,在阅读@brit的评论后,这是检查该错误的记录方法。请参阅 的文档:
如果超过截止日期,则调用 read 或 write 或其他 i/o 方法将返回一个包含 os.errdeadlineexceeded 的错误。 这可以使用 error.is(err, os.errdeadlineexceeded) 进行测试。
另一个表明错误是“超时”错误的迹象是它是否有 timeout() bool
方法,该方法在调用时返回 true
。
这是 接口的一部分(尽管该接口有一个额外的方法,已被记录为已弃用),并由 类型(以及内部 类型)实现。
有几个函数(例如 、 和公共函数 )可以通过直接将错误值转换为 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() }
请注意:
上面的 istimeout()
函数和 errors.is(err, os.errdeadlineexceeded)
调用之间的区别在于后者会尝试精确匹配 errdeadlineexceeded
(这主要是通过设置 setdeadline()
触发的)上一些类似文件或连接的对象),而后一个可能会因为其他原因尝试宣传“我是超时错误”的错误返回 true(例如:http“408 请求超时”响应)
注意:以上所有链接均指 go 1.18.3 。根据您使用的 go 版本,您可能需要调整上面的一些代码(例如:在 中添加了 os.errdeadlineexceeded
)
终于介绍完啦!小伙伴们,这篇关于《处理网络数据包读取超时后的错误》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~公众号也会发布Golang相关知识,快来关注吧!