http://www.neversaydie.cc/windows-and-linux-socket-port-reuse-characteristics-are-not-the-same/
昨天在ubuntu/linux下测试的时候,启动java程序报错“Failed to bind to”绑定端口失败,如下图
我首先怀疑是端口被其他程序占用了,关闭程序后查看了下端口信息
1
2
3
|
ps
-ef |
grep
java
kill
-9 <pid>
netstat
-apn |
grep
<port>
|
结果发现该端口没有被占用。
然后我怀疑的java程序本身的问题,再次kill掉程序后重新启动,等它报错后查看端口信息,如下图
占用该端口的就是这个java程序本身,可是为什么它还会报错?我把java程序复制一份,放到windows下运行,诡异的事情发生了,windows下居然没有报错。
最后我们准备调试程序,在ubuntu上下载了一个eclipse,安装maven,导入项目,让程序员debug一下。然后找到问题的原因了
我们可以看到,一个端口在程序里绑定了两次!!!可是为什么在windows下没有报错呢? 我上网查询了资料,现在整理如下:
每个TCP连接都是通过它的本地IP,本地端口及远程IP,远程端口组合,“独一无二”地标识出来的。
linux下,两个tcp的socket不能绑定同一个端口;而如果使用SO_REUSEADDR选项,两个udp的socket可以绑定同一个端口。
freebsd下,两个tcp的socket绑定同一端口,只有第一个socket获得数据。
windows下 两个tcp的socket不能绑定同一个端口,唯一例外的是监听socket。
图2中可以看出socket的类型的LISTEN,所以可以在windows下可以绑定多次而不报错。