socket.error: [Errno 98] Address already in use

来源:互联网 发布:猎豹浏览器有mac版吗 编辑:程序博客网 时间:2024/05/15 13:04

最近在学Karrigell的时候,终止了服务,再运行python Karrigell的时候,出现了socket.error: [Errno 98] Address already in use

这个错误,在网上查了一下原来是当通过客户端向服务器端发送消息,关闭了连接后,这时如果马上再去运行服务器端程序,会提示这个错误。

 

这与TCP/IP协议终止连接时候发出ACK之后需要等2个MSL时间有关,我们先看一下TCP/IP协议的三次握手和四次挥手:

三次握手(Three-Way Handshake)即建立TCP连接,就是指建立一个TCP连接时,需要客户端和服务端总共发送3个包以确认连接的建立。在socket编程中,这一过程由客户端执行connect来触发,整个流程如下图所示:


图2 TCP三次握手

        1.第一次握手:Client将标志位SYN置为1,随机产生一个值seq=J,并将该数据包发送给Server,Client进入SYN_SENT状态,等待Server确认。
         2.第二次握手:Server收到数据包后由标志位SYN=1知道Client请求建立连接,Server将标志位SYN和ACK都置为1,ack=J+1,随机产生一个值seq=K,

           并将该数据包发送给Client以确认连接请求,Server进入SYN_RCVD状态。
         3.第三次握手:Client收到确认后,检查ack是否为J+1,ACK是否为1,如果正确则将标志位ACK置为1,ack=K+1,并将该数据包发送给Server,

           Server检查ack是否为K+1,ACK是否为1,如果正确则连接建立成功,Client和Server进入ESTABLISHED状态,完成三次握手,随后Client与Server

           之间可以始传输数据了。

 

 

四次挥手(Four-Way Wavehand)即终止TCP连接,就是指断开一个TCP连接时,需要客户端和服务端总共发送4个包以确认连接的断开。在socket编程中,这一过程由客户端或服务端任一方执行close来触发,整个流程如下图所示:


图3 TCP四次挥手

        由于TCP连接时全双工的,因此,每个方向都必须要单独进行关闭,这一原则是当一方完成数据发送任务后,发送一个FIN来终止这一方向的连接,收

         到一个FIN只是意味着这一方向上没有数据流动了,即不会再收到数据了,但是在这个TCP连接上仍然能够发送数据,直到这一方向也发送了FIN。首

         先进行关闭的一方将执行主动关闭,而另一方则执行被动关闭,上图描述的即是如此。
         1.第一次挥手:Client发送一个FIN,用来关闭Client到Server的数据传送,Client进入FIN_WAIT_1状态

         2.第二次挥手:Server收到FIN后,发一个ACK给Client,确认序号为收到序号+1(与SYN相同,一个FIN占用一个序号),Server进入CLOSE_WAIT状态
         3.第三次挥手:Server发送一个FIN,用来关闭Server到Client的数据传送,Server进入LAST_ACK状态

         4.第四次挥手:Client收到FIN后,Client进入TIME_WAIT状态,接着发送一个ACK给Server,确认序号为收到序号+1Server进入CLOSED状态,完成

           四次挥手。
        上面是一方主动关闭,另一方被动关闭的情况,实际中还会出现同时发起主动关闭的情况,具体流程如下图:


图4 同时挥手

 

 

    在TCP/IP终止连接的四次挥手中,当最后的ACK回复发出后,有个2MSL的时间等待,MSL指一个片段在网络中最大的存活时间,这个时间一般是30秒,所以基本上过60秒后
 
    就可以重新连接!为什么要等待2MSL?是因为在最后发出ACK回复后,发送方不能确认ACK是否被另一端正常收到,如果另一端没有收到ACK回复的话,将会在1MSL后再
 
   次发送FIN片段。所以说发送方等待2MSL时间,也就是刚好它发ACK回复和对方发送FIN片段的时间,如果此时间内都没有再次收到FIN片段的话,发送方就假设对方已经正常
 
    接收到了ACK回复,此时它就会正常关闭连接!
 
 
    所以解决的方法就是: 使用setsockopt,使得socket可以被重用,是最常用的服务器编程要点。具体的做法为是,在socket调用和bind调用之间加上一段对
 
   socket的设置。

 

 

        sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)          sock.settimeout(CHECK_TIMEOUT)        sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)          sock.bind(('', UDP_PORT))  


这里涉及setsockopt函数:参考http://blog.chinaunix.net/uid-14966892-id-2780357.html

 

 


0 0