我想在golang中捕获“bind:address has in use”错误。
conn, err := net.ListenUDP("udp", addr)
if err != nil {
if CATCH_BIND_ERROR(err) {
// Do something if 'addr' is already in use
} else {
panic(err)
}
}
有没有办法实现CATCH_BIND_ERROR功能?
简单的方法是检查错误的文本:
conn, err := net.ListenUDP("udp", addr)
if err != nil && err.Error() == "bind: address already in use" {
// Failed to bind, do something
}
if err != nil {
// Some other error
panic(err)
}
对于这样的简单情况,这可能就足够了。然而,这种方法有点脆弱:
为了缓解这些问题,您可以检查特定的错误类型,而不仅仅是文本表示。在您的示例中,net包提供了一些自定义错误。 ListenUDP
方法返回net.OpError
,这意味着您可以更仔细地检查它。例如:
conn, err := net.ListenUDP("udp", addr)
if opErr, ok := err.(*net.OpError); ok {
if opErr.Op == "listen" && strings.Contains(opErr.Error.Error(), "address already in use") {
// Failed to bind, do something
}
}
if err != nil {
// Some other error, panic
panic(err)
}
在这种情况下,我们仍然依赖于文本检查,因此未来库更改仍有可能破坏测试。但是通过检查net.OpError
类型,我们确实减轻了第二个风险,可能会引发一些具有相同文本的其他错误。
在Windows上,错误消息是“Only one usage of each socket address (protocol/network address/port) is normally permitted.
”
此外,在本地化的情况下,消息将改变。
以下是捕获“已使用地址”错误的可能解决方案:
func isErrorAddressAlreadyInUse(err error) bool {
errOpError, ok := err.(*net.OpError)
if !ok {
return false
}
errSyscallError, ok := errOpError.Err.(*os.SyscallError)
if !ok {
return false
}
errErrno, ok := errSyscallError.Err.(syscall.Errno)
if !ok {
return false
}
if errErrno == syscall.EADDRINUSE {
return true
}
const WSAEADDRINUSE = 10048
if runtime.GOOS == "windows" && errErrno == WSAEADDRINUSE {
return true
}
return false
}
要使用此功能:
conn, err := net.ListenUDP("udp", addr)
if err != nil {
if isErrorAddressAlreadyInUse(err) {
// Do something if 'addr' is already in use
} else {
panic(err)
}
}