我们的Java程序之一在启动时仅监听IPv6(8080)
例如
# netstat -ntpl Proto Recv-Q Send-Q Local Address Foreign Address State PID/Program name tcp 0 0 0.0.0.0:22 0.0.0.0:* LISTEN - tcp6 0 0 :::8080 :::* LISTEN - tcp6 0 0 :::22 :::* LISTEN -
问题是无法从外部访问(本地主机除外),要解决此问题,我需要手动添加
-Djava.net.preferIPv4Stack=true
但这使得该程序仅适用于IPv4网络。
是否可以执行上述类似 sshd的 操作,并且都支持IPv4和IPv6?
我怀疑这不是Java编程问题,而是OS网络堆栈/ OS网络配置问题:
http://coding.derkeiler.com/Archive/Java/comp.lang.java.help/2009-09/msg00087.html 在某些操作系统上,单个本机TCP套接字可以同时侦听IPv4和IPv6上的端口。它能够接受来自远程IPv4和远程IPv6客户端的连接。在其他操作系统(例如WinXP)上,操作系统本机套接字无法执行此操作,但只能从IPv4或IPv6接受,不能同时接受。在这些OS上,必须具有两个侦听套接字才能接收来自远程IPv4和IPv6客户端的连接,一个套接字侦听IPv4连接,一个套接字侦听IPv6。
http://coding.derkeiler.com/Archive/Java/comp.lang.java.help/2009-09/msg00087.html
在某些操作系统上,单个本机TCP套接字可以同时侦听IPv4和IPv6上的端口。它能够接受来自远程IPv4和远程IPv6客户端的连接。在其他操作系统(例如WinXP)上,操作系统本机套接字无法执行此操作,但只能从IPv4或IPv6接受,不能同时接受。在这些OS上,必须具有两个侦听套接字才能接收来自远程IPv4和IPv6客户端的连接,一个套接字侦听IPv4连接,一个套接字侦听IPv6。
Windows 7和Windows Server 2008可以很好地处理双堆栈。Windows XP没有那么多:)
您似乎在Linux上-大多数现代Linux台式机和服务器也可以毫无问题地处理双ipv4 ipv6。
这是一篇关于互操作性的好文章:
您知道如何为您的Java应用程序“关闭” IPV6: -Djava.net.preferIPv4Stack=true
您还可以像这样强制服务器使用IPV6: echo 0 > /proc/sys/net/ipv6/bindv6only
echo 0 > /proc/sys/net/ipv6/bindv6only
可以说这是您的最佳来源:
除非受到外部网络问题的限制,否则您应该绝对能够完成所需的工作(至少在Java编程级别)。
Nodes) V4 Only V4/V6 V6 Only ------- ----- ------- V4 Only x x V4/V6 x x x V6 Only x x
PS:
这里还有一个很好的链接,它解释了套接字级别发生的情况。它不是Java(它是C),但实际上适用了示例原则: