假如有两个脚本,foo.py,bar.py,分别运行,都可以正常运行。但是现在想从两个程序间传递数据。
C/S 即 Client 和Server —>客户端和服务器端架构
B/S 即 Browser 和 Server —> 浏览器端和服务器端架构
网络就是一种辅助双方或者多方能够连接在一起的工具
伴随着网络发展,人们使用了很多通信方法,有些已不再使用,现在使用最广泛的是TCP/IP(Transmission Control Protocol/Internet Protocol)
TCP、IP 是标准的协议,其可以使世界范围内的计算机通过Internet 或本地网络通信。
TCP/IP事实上是一些协议(protocols)的合集。当前大多数使用中的通信都使用TCP协议。
Internet是在一些共享的线路上发送数据’的。例如:在您的计算机上也许同时运行着几个应用程序,如Web浏览器、通讯软件等程序,而您只须通过一条单一的线路来连接互联网。上面所有的程序都共享这个连接,简单地说,用户往往不会觉察到这个共享的发生。
让不同电脑中的软件能够进行数据传递,即网络中不同主机进程间的通信
不同的网络中,采用唯一的标识来区分不同的主体,比如车牌号、建筑物地址、电话号码、员工编号等等
一台电脑怎么找到很多电脑中的一台呢?
警察怎么找到嫌疑犯的?
ip地址:用来在网络中标记一台电脑,比如192.168.1.1;在同一网络中是唯一的。
同一网络:好比班级编号,球队编号。
同一网络:例如同一个局域网, 一个教室里。
目前IP主要分为两种
每一个IP包含两部分:
类似电话号码由区号+电话主机号组成
[外链图片转存失败(img-FCp0Yb5N-1565138244109)(/Users/xujiaqi/Library/Application Support/typora-user-images/image-20190805094830739.png)]
(1) IPv4可提供4,294,967,296个地址,IPv6将原来的32位地址空间增大到128位,数目是2的128次方。能够对地球上每平方米提供6×1023个网络地址,在可预见的将来是不会耗尽的。
(2) IPv4 使用地址解析通讯协议 (ARP) ,IPv6使用用多点传播 Neighbor Solicitation 消息取代地址解析通讯协议 (ARP) 。
(3) IPv4 中路由器不能识别用于服务质量的QoS 处理的 payload。IPv6中路由器使用 Flow Label 字段可以识别用于服务质量的 QoS 处理的 payload。
(4) IPv4的回路地址为: 127.0.0.1,IPv6的回路地址为 : 000:0000:0000:0000:0000:0000:0000:0001 可以简写为 ::1。
(5) 在IPv4中,动态主机配置协议( Dynamic Host ConfigurationProtocol,DHCP)实现了主机IP地址及其相关配置的自动设置。一个DHCP服务器拥有一个IP地址池,主机从DHCP服务器租借IP地址并获得有关的配置信息(如缺省网关、DNS服务器等),由此达到自动设置主机IP地址的目的。IP v6继承了IPv4的这种自动配置服务,并将其称为全状态自动配置(stateful autoconfiguration)。 m.pcwenku.com 供稿
(6) IPv4使用 Internet 群组管理通讯协议 (IGMP) 管理本机子网络群组成员身份,IPv6使用 Multicast Listener Discovery (MLD) 消息取代 IGMP。
(7) 内置的安全性。IPSec由IETF开发是确保秘密、完整、真实的信息穿越公共IP网的一种工业标准。IPsec不再是IP协议的补充部分,在IPv6中IPsec是IPv6自身所具有的功能。IPv4选择性支持IPSec,IPv6自动支持IPSec。
(8) 更好的QoS支持。QoS是网络的一种安全机制,通常情况下不需要QoS,但是对关键应用和多媒体应用就十分必要。当网络过载或拥塞时,QoS 能确保重要业务量不受延迟或丢弃,同时保证网络的高效运行。在IPv6 的包头中定义了如何处理与识别传输, IPv6 包头中使用 Flow Label 来识别传输,可使路由器标识和特殊处理属于一个流量的封包。流量是指来源和目的之间的一系列封包,因为是在 IPv6 包头中识别传输,所以即使透过 IPSec 加密的封包 payload,仍可实现对 QoS 的支持。
一个A类IP地址由1字节的网络地址和3字节主机地址组成,网络地址的最高位必须是“0”,
地址范围1.0.0.1-126.255.255.254
二进制表示为:00000001 00000000 00000000 00000001 - 01111110 11111111 11111111 11111110
8位二进制全1就是255
可用的A类网络有126个,每个网络能容纳1677214个主机
一个B类IP地址由2个字节的网络地址和2个字节的主机地址组成,网络地址的最高位必须是“10”,
地址范围128.1.0.1-191.255.255.254
二进制表示为:10000000 00000001 00000000 00000001 - 10111111 11111111 11111111 11111110
可用的B类网络有16384个,每个网络能容纳65534主机
一个C类IP地址由3字节的网络地址和1字节的主机地址组成,网络地址的最高位必须是“110”
范围192.0.1.1-223.255.255.254
二进制表示为: 11000000 00000000 00000001 00000001 - 11011111 11111111 11111110 11111110
C类网络可达2097152个,每个网络能容纳254个主机
D类IP地址第一个字节以“1110”开始,它是一个专门保留的地址。
它并不指向特定的网络,目前这一类地址被用在多点广播(Multicast)中
多点广播地址用来一次寻址一组计算机
地址范围224.0.0.1-239.255.255.254
网上视频会议、网上视频点播就是采用多点广播
广播地址 (Broadcast Address) 是专门用于同时向同一网络中所有主机发送数据的一个地址。在使用TCP/IP 协议的网络中, 主机标识段 HOST ID 为全1 的IP 地址为广播地址,广播分组传送给 HOST ID 段所涉及的所有计算机. 例如, 对于 192.168.50.26(255.255.255.0) 网段,其广播地址为192.168.50.255, 当发出一个目的地址为 192.168.50.255 的数据时, 它将被分发给该网段上的所有计算机. 飞秋,内网通就是通过广播地址来广播数据的
以“1111”开始,为将来使用保留
E类地址保留,仅作实验和开发用
在这么多网络IP中,国际规定有一部分IP地址是用于我们的局域网使用,也就
是属于私网IP,不在公网中使用的,它们的范围是:
10.0.0.0~10.255.255.255
172.16.0.0~172.31.255.255
192.168.0.0~192.168.255.255
IP地址127.0.0.1~127.255.255.255用于回路测试,
如:127.0.0.1可以代表本机IP地址,用http://127.0.0.1
就可以测试本机中配置的Web服务器。
###进制转换
二进制基数:0,1 以ob开头
八进制:0-7 以O开头
十进制:0-9 默认
十六进制:0-9,ABCDEF(0-15) 以OX开头
二进制 | 八进制 | 十进制 | 十六进制 |
---|---|---|---|
0000 0001 | 0000 0001 | 1 | 0000 0001 |
0000 0010 | 0000 0002 | 2 | 0000 0002 |
0000 0011 | 0000 0003 | 3 | 0000 0003 |
0000 0100 | 0000 0004 | 4 | 0000 0004 |
0000 0101 | 0000 0005 | 5 | 0000 0005 |
0000 0110 | 0000 0006 | 6 | 0000 0006 |
0000 0111 | 0000 0007 | 7 | 0000 0007 |
0000 1000 | 0000 0010 | 8 | 0000 0008 |
0000 1001 | 0000 0011 | 9 | 0000 0009 |
0000 1010 | 0000 0012 | 10 | 0000 000A |
0000 1011 | 0000 0013 | 11 | 0000 000B |
0000 1100 | 0000 0014 | 12 | 0000 000C |
0000 1101 | 0000 0015 | 13 | 0000 000D |
0000 1110 | 0000 0016 | 14 | 0000 000E |
0000 1111 | 0000 0017 | 15 | 0000 000F |
####8421码
#####十进制二进制转换
Ob10011101 转十进制
128 64 32 16 8 4 2 1
1 0 0 1 1 1 0 1
1+4+8+16+128=157
136转二进制
128 64 32 16 8 4 2 1
1 0 0 0 1 0 0 0
136 = 128+8
把三位的二进制看出一位的八进制
Ob100 110
4 6
Ob010 010 011 001 100
2 2 3 1 4
7 6 5 4
Ob111 110 101 100
把四位二进制看成一位十六进制
Ob10 0110
2 6
Ob100 1111 0101
4 f 5
Ox f a 3 6 5
1111 1010 0011 0110 0101
- 如果两台电脑之间通过网线连接是可以直接通信的,但是需要提前设置好ip地址以及网络掩码
- 并且ip地址需要控制在同一网段内,例如 一台为
192.168.1.1
另一台为192.168.1.2
则可以进行通信
- 当有多台电脑需要组成一个网时,那么可以通过集线器(Hub)将其链接在一起
- 一般情况下集线器的接口较少
- 集线器有个缺点,它以广播的方式进行发送任何数据,即如果集线器接收到来自A电脑的数据本来是想转发给B电脑,如果此时它还连接着另外两台电脑C、D,那么它会把这个数据给每个电脑都发送一份,因此会导致网络拥堵
- 克服了集线器以广播发送数据的缺点,当需要广播的时候发送广播,当需要单播的时候又能够以单播的方式进行发送
- 它已经替代了之前的集线器
- 企业中就是用交换机来完成多态电脑设备的链接成网络的
较为复杂的通信过程如:www.baidu.com
- 在浏览器中输入一个网址时,需要将它先解析出ip地址来
- 当得到ip地址之后,浏览器以tcp的方式3次握手链接服务器
- 以tcp的方式发送http协议的请求数据 给 服务器
- 服务器tcp的方式回应http协议的应答数据 给浏览器
总结
MAC地址:在设备与设备之间数据通信时用来标记收发双方(网卡的序列号)
IP地址:在逻辑上标记一台电脑,用来指引数据包的收发方向(相当于电脑的序列号)
网络掩码:用来区分ip地址的网络号和主机号
默认网关:当需要发送的数据包的目的ip不在本网段内时,就会发送给默认的一台电脑,成为网关
集线器:已过时,用来连接多态电脑,缺点:每次收发数据都进行广播,网络会变的拥堵
交换机:集线器的升级版,有学习功能知道需要发送给哪台设备,根据需要进行单播、广播
路由器:连接多个不同的网段,让他们之间可以进行收发数据,每次收到数据后,ip不变,但是MAC地址会变化
DNS:用来解析出IP(类似电话簿)
http服务器:提供浏览器能够访问到的数据
开放式系统互联参考模型(Open System Interconnection Reference Model)—>简称OSI
这个标准定义了网络的七层框架,试图使得计算机在整个世界范围内实现互联。
在OSI中,网络体系结构被分成下面的七层。
按照分工不同把互联网协议从逻辑上划分了层级:
[外链图片转存失败(img-BzjpkHPT-1565138244110)(/Users/xujiaqi/Library/Application Support/typora-user-images/image-20190805102546169.png)]
整个网络通信通过IP地址+端口标示不同的网络服务
端口号是用来表示区别网络中不同的应用,操作系统会对端口进行编号,即端口号
端口的分配是基于一定规则的,而不是随机分配的
知名端口
80,分配给http服务的
21,分配给FTP服务
动态端口(Dynamic Ports)
一般不固定分配某种服务,动态分配,范围:1024-65535
所谓的动态分配,是指一个程序需要网络通信时,它向主机申请一个端口,主机从可用的端口号分配一个供其使用。关闭程序时,同时释放占用的端口
端口查看
1、查看进程号
ps -ef | grep 进程名
2、查看端口被哪个进程监听
sudo lsof -i :端口
3、查看进程监听的端口
sudo lsof -nP -p 进程号 | grep LISTEN
sudo lsof -nP | grep LISTEN | grep 进程号
4、查看监听端口的进程
sudo lsof -nP | grep LISTEN | grep 端口号
不同的电脑上进程的标识与识别
用唯一标识来标记一个进程
在电脑上,可以通过用进程号(PID)来唯一标识进程,但是在网络上,不可以,需要利用TCP/IP协议族来帮我们解决问题
用IP可以唯一标识网络中的主机,协议+端口号唯一标识主机中的应用进程
进程指的是,运行的程序以及运行程序用到的资源的整体就称之为进程
Socket(套接字)是进程间通信的一种方式
与其他进程通信的一个主要不同是:
它能不同主机间的进程间的通信
现行大多数服务都是基于Socket来完成通信的
比如:浏览网页,QQ聊天,收发Email
import socket
socket.socket(AddressFamily,Type)
参数说明:
创建一个TCP Socket
import socket
# 创建一个socket对象
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# print(dir(s))
# 获取一个本地的主机名
host = socket.gethostname()
# 设置端口号
port = 1234
# 连接服务器,指定主机和端口
s.connect((host, port))
# 接受小于1024字节的数据
msg = s.recv(1024)
s.close()
print(msg.decode('utf-8'))
import socket
# 创建一个socket对象
serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 主机名
host = socket.gethostname()
# 端口号
port = 1234
# 绑定端口
serversocket.bind((host, port))
# 设置最大连接数,超过后排队
serversocket.listen(5)
while True:
# 建立客户端连接
clientsocket, addr = serversocket.accept()
print('连接地址:\t %s'%(str(addr)))
msg = '网络编程测试。'+'\r\n'
clientsocket.send(msg.encode('utf-8'))
clientsocket.close()
out:
连接地址: ('192.168.1.136', 52611)
UDP—>User Datagram Protocol(用户数据报协议)是一个无连接简单的面向数据报的运输层协议。
优缺点:
优点
缺点
特点:
UDP是面向无连接的通讯协议
UDP包括目的的端口号和源端口号信息,通讯不需要连接,能够实现广播发送
UDP数据大小有限制,每个被传输的数据报必须限定在64k以内
UDP是一个不可靠的协议,发送出去的数据报不一定以相同次序到达接收方
UDP一般多用于多点通信和实时的数据业务,比如:
创建一个UDP客户端程序的流程:
1.创建一个客户端套接字
2.发送/接受数据
3.关闭套接字
Socket 和file 的区别
import socket
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
addr = ('192.168.1.136', 1234)
sendData = input('其请输入要发送的数据:')
udp_socket.sendto(sendData.encode('gbk'), addr)
udp_socket.close()!image-20190805151723236](/Users/xujiaqi/Desktop/image-20190805151723236.png)
[外链图片转存失败(img-5TjasO6j-1565138244111)(/Users/xujiaqi/Library/Application Support/typora-user-images/image-20190805151833049.png)]
import socket
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
Addr = ('192.168.1.112', 8345)
sendData = input('请输入要发送的数据:')
udp_socket.sendto(sendData.encode('utf-8'), Addr)
recvData = udp_socket.recvfrom(1024)
print(recvData)
udp_socket.close()
函数 | 描述 |
---|---|
服务器端套接字 | |
s.bind() | 绑定地址(host,port)到套接字, 在AF_INET下,以元组(host,port)的形式表示地址。 |
s.listen() | 开始TCP监听。backlog指定在拒绝连接之前,操作系统可以挂起的最大连接数量。该值至少为1,大部分应用程序设为5就可以了。 |
s.accept() | 被动接受TCP客户端连接,(阻塞式)等待连接的到来 |
客户端套接字 | |
s.connect() | 主动初始化TCP服务器连接,。一般address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。 |
s.connect_ex() | connect()函数的扩展版本,出错时返回出错码,而不是抛出异常 |
公共用途的套接字函数 | |
s.recv() | 接收TCP数据,数据以字符串形式返回,bufsize指定要接收的最大数据量。flag提供有关消息的其他信息,通常可以忽略。 |
s.send() | 发送TCP数据,将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。 |
s.sendall() | 完整发送TCP数据,完整发送TCP数据。将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。 |
s.recvform() | 接收UDP数据,与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。 |
s.sendto() | 发送UDP数据,将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。 |
s.close() | 关闭套接字 |
s.getpeername() | 返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。 |
s.getsockname() | 返回套接字自己的地址。通常是一个元组(ipaddr,port) |
s.setsockopt(level,optname,value) | 设置给定套接字选项的值。 |
s.getsockopt(level,optname[.buflen]) | 返回套接字选项的值。 |
s.settimeout(timeout) | 设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。一般,超时期应该在刚创建套接字时设置,因为它们可能用于连接的操作(如connect()) |
s.gettimeout() | 返回当前超时期的值,单位是秒,如果没有设置超时期,则返回None。 |
s.fileno() | 返回套接字的文件描述符。 |
s.setblocking(flag) | 如果flag为0,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用recv()没有发现任何数据,或send()调用无法立即发送数据,那么将引起socket.error异常。 |
s.makefile() | 创建一个与该套接字相关连的文件 |
用网络调试助手时,端口号会一直变动。
一般情况,一台电脑运行的网络程序有很多,为了不与其他的网络程序占用同一个端口号,往往在编程中,udp的端口号一般不绑定
但是如果需要做成一个服务器端的程序的话,是需要绑定的,想想看这又是为什么呢?
如果报警电话每天都在变,想必世界就会乱了,所以一般服务性的程序,往往需要一个固定的端口号,这就是所谓的端口绑定
import socket
udp_socket = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
local_addr = ('', 7788) # ip地址和端口号,IP一般不用写,表示本地ip
udp_socket.bind(local_addr)
recv_data = udp_socket.recvfrom(1024)
print(recv_data[0].decode('gbk'))
udp_socket.close()
总结:
一个udp网络程序,可以不绑定端口,此系统会自动分配一个端口,重新运行此程序,端口号可能会发生变化
一个udp网络程序,可以绑定信息(IP,Ports),如果绑定成功,那么操作系统用这个端口号来进行区别收到的网络数据是否是此进程的。
TCP协议,传输控制协议(Transmission Control Protocol),是一种面向连接的,可靠的,基于字节流的传输层通信协议。
TCP通信需要经过创建连接,传输数据,终止连接三个步骤。
通信双方必须先建立连接才能进行数据的传输,双方都必须为该连接分配必要的系统内核资源,以管理连接的状态和连接上的传输。
双方间的数据传输都可以通过这个连接进行。
完成数据交换后,双方断开此连接,以释放系统资源。
这种连接是一对一的
因此TCP不适用于广播的应用程序,基于广播的应用程序请使用UDP协议。
1)TCP采用发送应答机制
TCP发送的每个报文段都必须得到接收方的应答才认为这个TCP报文段传输成功
2)超时重传
发送端发出一个报文段之后就启动定时器,如果在定时时间内没有收到应答就重新发送这个报文段。
TCP为了保证不发生丢包,就给每个包一个序号,同时序号也保证了传送到接收端实体的包的按序接收。然后接收端实体对已成功收到的包发回一个相应的确认(ACK);如果发送端实体在合理的往返时延(RTT)内未收到确认,那么对应的数据包就被假设为已丢失将会被进行重传。
3)错误校验
TCP用一个校验和函数来检验数据是否有错误;在发送和接收时都要计算校验和。
流量控制用来避免主机发送得过快而使接收方来不及完全收下。
在Python中,用TCP进行socket编程也比较简单
通常,服务器程序可以无限运行下去。要注意的是,一个端口不能同时被两个socket绑定。
TCP服务器和客户端格子socket的创建和交互。
[外链图片转存失败(img-1cQ1ttPX-1565138244111)(/Users/xujiaqi/Library/Containers/com.tencent.qq/Data/Library/Caches/Images/438008605F7EBB108D567A53183CA90A.png)]
[外链图片转存失败(img-IUedNgBj-1565138244112)(/Users/xujiaqi/Downloads/852422-20161205000926693-716693816.png)]
在Python程序中,如果要完成一个TCP服务器的功能,需要流程如下:
服务器端:
import socket
# 创建套接字
tcp_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 绑定本地信息
tcp_socket.bind(('', 8080)) # port num>1024的随意用,<1024的一般指定类用途
# 设置监听
tcp_socket.listen(5)
#创建新的套接字
new_socket,addr = tcp_socket.accept()
#收/发数据
content = new_socket.recv(1024)
print('接收到的数据:',content.decode('gbk')) #解码
data = input('服务器发送的数据:')
new_socket.send(data.encode('gbk')) #编码为一个二进制的数
#关闭通信的socket
new_socket.close()
#关闭用于监听的套接字
tcp_socket.close()
设置监听的目的:
socket默认为主动模式,listen()能够将主动模式改为模式,被动才能接收信息
listen(5),5是指可以同时接收到的客户端申请的最大数,超过则拒绝连接
创建新套接字的原因:
服务器没收到请求后,将生成一个新的套接字,专门给新来的客户端提供一对一的服务
此时,服务器套接字的任务就是,等待新的客户端套接字连接它
客户端:
import socket
# 创建套接字
tcp_client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 目标服务器信息
server_ip = input('请输入服务器IP:')
server_port = int(input('请输入服务器port:'))
# 连接服务器
tcp_client_socket.connect((server_ip, server_port))
# 提示用户输入数据
send_data = input('请输入要发送的数据:')
tcp_client_socket.send(send_data.encode('utf-8'))
#接收服务器端发送来的信息
recv_data = tcp_client_socket.recv(1024)
print('接收到的数据为:',recv_data.decode('gbk'))
#关闭套接字
tcp_client_socket.close()
TCP协议
当应用程序希望通过TCP与另一个应用程序通信时,他会发送一个通信请求,这个请求必须被送到一个确切的地址,在双方’握手’之后,TCP将在两个应用程序之间建立一个全双工(full-duplex)的通信,这个全双工的通信将占用两个计算机之间的通信线路,直到他被一方或者双方关闭为止
标志位:TCP的6个标志位
所谓标志位,一种二进制的指令
####TCP三次握手
Tcp:是因特网的传输层协议,使用三次握手协议简历连接,当主动发出SYN请求后,等待对方的回答SYN+ACK,并最终对对方的SYN执行ACK确认。这种建立连接的方法可以防止产生错误的连接
TCP三次握手的过程
三次握手完成,TCP客户端和服务器客户端成功过建立连接,可以开始传输数据
####TCP四次握手
建立一个连接需要三次握手,而终止一个连接要经过四次握手,这个是由TCP的半关闭(Half-close)造成的
某个应用进程首先调用close,称该端执行’主动关闭’(active close)该端的TCP(tool center point)于是发送了一个FIN字节,表示数据发送完毕
接收到这个FIN的对端执行"被动关闭"(passive close),这个FIN由TCP确认。
注意:
FIN的接收也作为一个文件结束符(end-of-file)传递给接收端应用进程,放在排队等候该应用进程接收的任何其他数据之后,因为,FIN接收意味着接收端应用进程在相关的连接上再无额外数据可接收
一段时间后,接收到这个文件结束符的应用,进程将会调用close关闭它的套接字。这导致它的TCP也发送一个FIN
接收这个最终FIN的原发送端TCP(即执行主动关闭的那一端)确认这个FIN,(既然每端都需要一个FIN和一个ACK,因此,通常需要四个字节)
假设Client端发起中断连接请求,也就是发送FIN报文。Server端接到FIN报文后,意思是说"我Client端没有数据要发给你了",但是如果你还有数据没有发送完成,则不必急着关闭Socket,可以继续发送数据。所以你先发送ACK,“告诉Client端,你的请求我收到了,但是我还没准备好,请继续你等我的消息”。这个时候Client端就进入FIN_WAIT状态,继续等待Server端的FIN报文。当Server端确定数据已发送完成,则向Client端发送FIN报文,“告诉Client端,好了,我这边数据发完了,准备好关闭连接了”。Client端收到FIN报文后,"就知道可以关闭连接了,但是他还是不相信网络,怕Server端不知道要关闭,所以发送ACK后进入TIME_WAIT状态,如果Server端没有收到ACK则可以重传。“,Server端收到ACK后,“就知道可以断开连接了”。Client端等待了2MSL后依然没有收到回复,则证明Server端已正常关闭,那好,我Client端也可以关闭连接了。Ok,TCP连接就这样关闭了!
[外链图片转存失败(img-gnXL2DYk-1565138244112)(/Users/xujiaqi/Downloads/image.png)]