代码
#!/usr/bin/python
import errno
import socket
import sys
import time
if len(sys.argv) < 2:
print "Usage: %s port" % sys.argv[0]
print "port should in net.ipv4.ip_local_port_range"
else:
port = int(sys.argv[1])
for i in range(65536):
try:
sock = socket.create_connection(('localhost', port))
print "connected", sock.getsockname(), sock.getpeername()
time.sleep(60*60)
except socket.error, e:
if e.errno != errno.ECONNREFUSED:
break
实验
netstat -ltnp
//可看到22号在监听
[root@localhost python]# python self-cnnect.py 22
结果如下
[root@localhost python]# python self-connect.py 22
connected ('::1', 58770, 0, 0) ('::1', 22, 0, 0)
下面连接一个35530(并没有监听),也会连接成功
[dongh@dl1 ~]$ python self-connect.py 35530
connected ('::1', 35530, 0, 0) ('::1', 35530, 0, 0)
看下本机端口号的范围
[root@localhost python]# sysctl -A | grep range
net.ipv4.ip_local_port_range = 32768 61000
为什么会出现自连接(自会在本机出现)
因为TCP在发起一个连接的时候会选着一个端口号(从ip_local_port_range),不是随机选,而是大概是一个计数器的方式,这次是x,下一个是x+1;选定后向服务器端口发送一个SYN请求;
如果服务器有进程在监听此端口,这客户端是不可能选到此端口的;没有在监听的情况下,在选好端口和IP的时候,这一项已经在在内核当中了,(上 例子IP 127.0.0.1 端口35530,则连接的时候客户端向IP 127.0.0.1 端口35530的服务端发送,而内核中有此项,则可以建立连接,单如果向IP 127.0.0.1 端口34444,而内核没有打开此项,则会返回一个RST)
如何解决呢?
网络库应该很容易判断,判断本机IP地址是否等于对方IP地址即可。