zoukankan      html  css  js  c++  java
  • TCP连接状态具体解释

    tcp状态:

    LISTEN:侦听来自远方的TCPport的连接请求

    SYN-SENT:再发送连接请求后等待匹配的连接请求

    SYN-RECEIVED:再收到和发送一个连接请求后等待对方对连接请求的确认

    ESTABLISHED:代表一个打开的连接

    FIN-WAIT-1:等待远程TCP连接中断请求,或先前的连接中断请求的确认

    FIN-WAIT-2:从远程TCP等待连接中断请求

    CLOSE-WAIT:等待从本地用户发来的连接中断请求

    CLOSING:等待远程TCP对连接中断的确认

    LAST-ACK:等待原来的发向远程TCP的连接中断请求的确认

    TIME-WAIT:等待足够的时间以确保远程TCP接收到连接中断请求的确认

    CLOSED:没有不论什么连接状态

    TCP是一个面向连接的协议,所以在连接两方发送数据之前,都须要首先建立一条连接。这和前面讲到的协议全然不同。前面讲的全部协议都仅仅是发送数据而已,大多数都不关心发送的数据是不是送到,UDP尤其明显,从编程的角度来说,UDP编程也要简单的多----UDP都不用考虑数据分片。

    书中用telnet登陆退出来解释TCP协议连接的建立和中止的过程,能够看到,TCP连接的建立能够简单的称为三次握手,而连接的中止则能够叫做四次握手

    1.连接的建立

    在建立连接的时候,client首先向server申请打开某一个port(用SYN段等于1的TCP报文),然后server端发回一个ACK报文通知client请求报文收到,client收到确认报文以后再次发出确认报文确认刚才server端发出的确认报文(绕口么),至此,连接的建立完毕。这就叫做三次握手。假设打算让两方都做好准备的话,一定要发送三次报文,并且仅仅须要三次报文就能够了。

    能够想见,假设再加上TCP的超时重传机制,那么TCP就全然能够保证一个数据包被送到目的地。

    2.结束连接

    TCP有一个特别的概念叫做half-close,这个概念是说,TCP的连接是全双工(能够同一时候发送和接收)连接,因此在关闭连接的时候,必须关闭传和送两个方向上的连接。客户机给server一个FIN为1的TCP报文,然后server返回给client一个确认ACK报文,而且发送一个FIN报文,当客户机回复ACK报文后(四次握手),连接就结束了。

    3.最大报文长度

    在建立连接的时候,通信的两方要互相确认对方的最大报文长度(MSS),以便通信。一般这个SYN长度是MTU减去固定IP首部和TCP首部长度。对于一个以太网,一般能够达到1460字节。当然假设对于非本地的IP,这个MSS可能就仅仅有536字节,并且,假设中间的传输网络的MSS更佳的小的话,这个值还会变得更小。

    4.TCP的状态迁移图

    书P182页给出了TCP的状态图,这是一个看起来比較复杂的状态迁移图,由于它包括了两个部分---server的状态迁移和client的状态迁移,假设从某一个角度出发来看这个图,就会清晰很多,这里面的server和client都不是绝对的,发送数据的就是client,接受数据的就是server。

    4.1.client应用程序的状态迁移图

    client的状态能够用例如以下的流程来表示:

    CLOSED->SYN_SENT->ESTABLISHED->FIN_WAIT_1->FIN_WAIT_2->TIME_WAIT->CLOSED

    以上流程是在程序正常的情况下应该有的流程,从书中的图中能够看到,在建立连接时,当client收到SYN报文的ACK以后,client就打开了数据交互地连接。而结束连接则一般是client主动结束的,client结束应用程序以后,须要经历FIN_WAIT_1,FIN_WAIT_2等状态,这些状态的迁移就是前面提到的结束连接的四次握手。

    4.2.server的状态迁移图

    server的状态能够用例如以下的流程来表示:

    CLOSED->LISTEN->SYN收到->ESTABLISHED->CLOSE_WAIT->LAST_ACK->CLOSED

    在建立连接的时候,server端是在第三次握手之后才进入数据交互状态,而关闭连接则是在关闭连接的第二次握手以后(注意不是第四次)。而关闭以后还要等待client给出最后的ACK包才干进入初始的状态。

    4.3.其它状态迁移

    书中的图另一些其它的状态迁移,这些状态迁移针对server和client双方面的总结例如以下

    1. LISTEN->SYN_SENT,对于这个解释就非常easy了,server有时候也要打开连接的嘛。
    2. SYN_SENT->SYN收到,server和client在SYN_SENT状态下假设收到SYN数据报,则都须要发送SYN的ACK数据报并把自己的状态调整到SYN收到状态,准备进入ESTABLISHED
    3. SYN_SENT->CLOSED,在发送超时的情况下,会返回到CLOSED状态。
    4. SYN_收到->LISTEN,假设受到RST包,会返回到LISTEN状态。
    5. SYN_收到->FIN_WAIT_1,这个迁移是说,能够不用到ESTABLISHED状态,而能够直接跳转到FIN_WAIT_1状态并等待关闭。

    4.4.2MSL等待状态

    书中给的图里面,有一个TIME_WAIT等待状态,这个状态又叫做2MSL状态,说的是在TIME_WAIT2发送了最后一个ACK数据报以后,要进入TIME_WAIT状态,这个状态是防止最后一次握手的数据报没有传送到对方那里而准备的(注意这不是四次握手,这是第四次握手的保险状态)。这个状态在非常大程度上保证了两方都能够正常结束,可是,问题也来了。

    因为插口的2MSL状态(插口是IP和port对的意思,socket),使得应用程序在2MSL时间内是无法再次使用同一个插口的,对于客户程序还好一些,可是对于服务程序,比如httpd,它总是要使用同一个port来进行服务,而在2MSL时间内,启动httpd就会出现错误(插口被使用)。为了避免这个错误,server给出了一个平静时间的概念,这是说在2MSL时间内,尽管能够又一次启动server,可是这个server还是要平静的等待2MSL时间的过去才干进行下一次连接。

    4.5.FIN_WAIT_2状态

    这就是著名的半关闭的状态了,这是在关闭连接时,client和server两次握手之后的状态。在这个状态下,应用程序还有接受数据的能力,可是已经无法发送数据,可是也有一种可能是,client一直处于FIN_WAIT_2状态,而server则一直处于WAIT_CLOSE状态,而直到应用层来决定关闭这个状态。

    5.RST,同一时候打开和同一时候关闭

    RST是还有一种关闭连接的方式,应用程序应该能够推断RST包的真实性,即是否为异常中止。而同一时候打开和同一时候关闭则是两种特殊的TCP状态,发生的概率非常小。

    6.TCPserver设计

    前面以前讲述过UDP的server设计,能够发现UDP的server全然不须要所谓的并发机制,它仅仅要建立一个数据输入队列就能够。可是TCP不同,TCPserver对于每个连接都须要建立一个独立的进程(或者是轻量级的,线程),来保证对话的独立性。所以TCPserver是并发的。并且TCP还须要配备一个呼入连接请求队列(UDPserver也相同不须要),来为每个连接请求建立对话进程,这也就是为什么各种TCPserver都有一个最大连接数的原因。而依据源主机的IP和port号码,server能够非常轻松的差别出不同的会话,来进行数据的分发。

  • 相关阅读:
    bash八大扩展一网打尽
    MySQL命令行导出数据库
    Windows 7上的DirectX 11.1
    把KlayGE嵌入其他GUI框架
    KlayGE的资源载入系统
    学习路漫漫……
    写下我的第一篇Post,呵呵
    今天学习:CSS中的类class和标识id选择符(.和#号)
    Remove Duplicates from Unsorted List
    2012 TODO List
  • 原文地址:https://www.cnblogs.com/bhlsheji/p/4259029.html
Copyright © 2011-2022 走看看