Python-网络编程(二)

一.网络通信原理python

  1.互联网的本质就是一系列的网络协议linux

  咱们是在浏览器上输入了一个网址,可是咱们都知道,互联网链接的电脑互相通讯的是电信号,咱们的电脑是怎么将咱们输入的网址变成了电信号而后发送出去了呢,而且咱们发送出去的消息是否是应该让对方的服务器可以知道,咱们是在请求它的网站呢,也就是说京东是否是应该知道我发送的消息是什么意思呢。是否是发送的消息应该有一些固定的格式呢?让全部电脑都能识别的消息格式,他就像英语成为世界上全部人通讯的统一标准同样,若是把计算机当作分布于世界各地的人,那么链接两台计算机之间的internet实际上就是一系列统一的标准,这些标准称之为互联网协议,互联网的本质就是一系列的协议,总称为‘互联网协议’(Internet Protocol Suite)。程序员

  互联网协议的功能:定义计算机如何接入internet,以及接入internet的计算机通讯的标准。算法

  网络通讯的流程昨天已经说过了,出门左转就能看到,这里就再也不说了.shell

  2.osi七层协议json

  互联网协议按照功能不一样分为osi七层或tcp/ip五层或tcp/ip四层windows

  咱们如今只须要了解五层的协议就行了,ok吗?咱们写的程序属于哪一层呢,属于应用层。设计模式

  每层运行常见物理设备浏览器

 

  3.tcp/ip五层模型讲解缓存

   咱们将应用层,表示层,会话层并做应用层,从tcp/ip五层协议的角度来阐述每层的由来与功能,搞清楚了每层的主要协议

    就理解了整个互联网通讯的原理。

    首先,用户感知到的只是最上面一层应用层,自上而下每层都依赖于下一层,因此咱们从最下一层开始切入,比较好理解

    每层都运行特定的协议,越往上越靠近用户,越往下越靠近硬件

     3.1 物理层

 

      物理层功能:主要是基于电器特性发送高低电压(电信号),高电压对应数字1,低电压对应数字0

    3.2 数据链路层

      数据链路层的功能:定义了电信号的分组方式

      以太网协议:

      早期的时候各个公司都有本身的分组方式,后来造成了统一的标准,即以太网协议ethernet

      ethernet规定

    一组电信号构成一个数据包,叫作‘帧’

    每一数据帧分红:报头head和数据data两部分

       mac地址:

        mac地址:每块网卡出厂时都被烧制上一个世界惟一的mac地址,长度为48位2进制,一般由12位16进制数表示(前六位是厂商编号,后六位是流水线号)

     3.3 网络层

      IP协议:

      规定网络地址的协议叫ip协议,它定义的地址称之为ip地址,普遍采用的v4版本即ipv4,它规定网络地址由32位2进制表示

      范围0.0.0.0-255.255.255.255 (4个点分十进制,也就是4个8位二进制数)

      一个ip地址一般写成四段十进制数,例:172.16.10.1

      ipv6,经过上面能够看出,ip紧缺,因此为了知足更多ip须要,出现了ipv6协议:6个冒号分割的16进制数表示,这个应该是未来的趋势,可是ipv4仍是用的最多的,由于咱们通常一个公司就一个对外的IP地址,咱们全部的机器上网都走这一个IP出口。

      ip数据包

      ip数据包也分为head和data部分,无须为ip包定义单独的栏位,直接放入以太网包的data部分

      head:长度为20到60字节

      data:最长为65,515字节。

      而以太网数据包的”数据”部分,最长只有1500字节。所以,若是IP数据包超过了1500字节,它就须要分割成几个以太网数据包,分开发送了。

    3.4 传输层

      tcp协议:(TCP把链接做为最基本的对象,每一条TCP链接都有两个端点,这种端点咱们叫做套接字(socket),它的定义为端口号拼接到IP地址即构成了套接字,例如,若IP地址为192.3.4.16 而端口号为80,那么获得的套接字为192.3.4.16:80。)

      当应用程序但愿经过 TCP 与另外一个应用程序通讯时,它会发送一个通讯请求。这个请求必须被送到一个确切的地址。在双方“握手”以后,TCP 将在两个应用程序之间创建一个全双工 (full-duplex,双方均可以收发消息) 的通讯。

      这个全双工的通讯将占用两个计算机之间的通讯线路,直到它被一方或双方关闭为止。

      它是可靠传输,TCP数据包没有长度限制,理论上能够无限长,可是为了保证网络的效率,一般TCP数据包的长度不会超过IP数据包的长度,以确保单个TCP数据包没必要再分割。

      udp协议:不可靠传输,”报头”部分一共只有8个字节,总长度不超过65,535字节,正好放进一个IP数据包。

      tcp三次握手和四次挥手

      咱们知道网络层,能够实现两个主机之间的通讯。可是这并不具体,由于,真正进行通讯的实体是在主机中的进程,是一个主机中的一个进程与另一个主机中的一个进程在交换数据。IP协议虽然能把数据报文送到目的主机,可是并无交付给主机的具体应用进程。而端到端的通讯才应该是应用进程之间的通讯。

      UDP,在传送数据前不须要先创建链接,远地的主机在收到UDP报文后也不须要给出任何确认。虽然UDP不提供可靠交付,可是正是由于这样,省去和不少的开销,使得它的速度比较快,好比一些对实时性要求较高的服务,就经常使用的是UDP。对应的应用层的协议主要有 DNS,TFTP,DHCP,SNMP,NFS 等。

      TCP,提供面向链接的服务,在传送数据以前必须先创建链接,数据传送完成后要释放链接。所以TCP是一种可靠的的运输服务,可是正由于这样,不可避免的增长了许多的开销,好比确认,流量控制等。对应的应用层的协议主要有 SMTP,TELNET,HTTP,FTP 等。

       三次握手:

      1.TCP服务器进程先建立传输控制块TCB,时刻准备接受客户进程的链接请求,此时服务器就进入了 LISTEN(监听)状态;

      2.TCP客户进程也是先建立传输控制块TCB,而后向服务器发出链接请求报文,这是报文首部中的同部位SYN=1,同时选择一个初始序列号 seq=x ,此时,TCP客户端进程进入了 SYN-SENT(同步已发送状态)状态。TCP规定,SYN报文段(SYN=1的报文段)不能携带数据,但须要消耗掉一个序号。

      3.TCP服务器收到请求报文后,若是赞成链接,则发出确认报文。确认报文中应该 ACK=1,SYN=1,确认号是ack=x+1,同时也要为本身初始化一个序列号 seq=y,此时,TCP服务器进程进入了SYN-RCVD(同步收到)状态。这个报文也不能携带数据,可是一样要消耗一个序号。

      4.TCP客户进程收到确认后,还要向服务器给出确认。确认报文的ACK=1,ack=y+1,本身的序列号seq=x+1,此时,TCP链接创建,客户端进入ESTABLISHED(已创建链接)状态。TCP规定,ACK报文段能够携带数据,可是若是不携带数据则不消耗序号。

      5.当服务器收到客户端的确认后也进入ESTABLISHED状态,此后双方就能够开始通讯了。 

      四次挥手:

      数据传输完毕后,双方均可释放链接。最开始的时候,客户端和服务器都是处于ESTABLISHED状态,而后客户端主动关闭,服务器被动关闭。服务端也能够主动关闭,一个流程。

      1.客户端进程发出链接释放报文,而且中止发送数据。释放数据报文首部,FIN=1,其序列号为seq=u(等于前面已经传送过来的数据的最后一个字节的序号加1),此时,客户端进入FIN-WAIT-1(终止等待1)状态。 TCP规定,FIN报文段即便不携带数据,也要消耗一个序号。

      2.服务器收到链接释放报文,发出确认报文,ACK=1,ack=u+1,而且带上本身的序列号seq=v,此时,服务端就进入了CLOSE-WAIT(关闭等待)状态。TCP服务器通知高层的应用进程,客户端向服务器的方向就释放了,这时候处于半关闭状态,即客户端已经没有数据要发送了,可是服务器若发送数据,客户端依然要接受。这个状态还要持续一段时间,也就是整个CLOSE-WAIT状态持续的时间。

      3.客户端收到服务器的确认请求后,此时,客户端就进入FIN-WAIT-2(终止等待2)状态,等待服务器发送链接释放报文(在这以前还须要接受服务器发送的最后的数据)。

      4.服务器将最后的数据发送完毕后,就向客户端发送链接释放报文,FIN=1,ack=u+1,因为在半关闭状态,服务器极可能又发送了一些数据,假定此时的序列号为seq=w,此时,服务器就进入了LAST-ACK(最后确认)状态,等待客户端的确认。

      5.客户端收到服务器的链接释放报文后,必须发出确认,ACK=1,ack=w+1,而本身的序列号是seq=u+1,此时,客户端就进入了TIME-WAIT(时间等待)状态。注意此时TCP链接尚未释放,必须通过2∗MSL(最长报文段寿命)的时间后,当客户端撤销相应的TCB后,才进入CLOSED状态。

      6.服务器只要收到了客户端发出的确认,当即进入CLOSED状态。一样,撤销TCB后,就结束了此次的TCP链接。能够看到,服务器结束TCP链接的时间要比客户端早一些。

 

    3.5 应用层

      应用层由来:用户使用的都是应用程序,均工做于应用层,互联网是开发的,你们均可以开发本身的应用程序,数据多种多样,必须规定好数据的组织形式 

      应用层功能:规定应用程序的数据格式。

  五层通讯流程:

 

二. socket

  结合上图来看,socket在哪一层呢,咱们继续看下图

  socket在内的五层通信流程:

  Socket又称为套接字,它是应用层与TCP/IP协议族通讯的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来讲,一组简单的接口就是所有,让Socket去组织数据,以符合指定的协议。当咱们使用不一样的协议进行通讯时就得使用不一样的接口,还得处理不一样协议的各类细节,这就增长了开发的难度,软件也不易于扩展(就像咱们开发一套公司管理系统同样,报帐、会议预约、请假等功能不须要单独写系统,而是一个系统上多个功能接口,不须要知道每一个功能如何去实现的)。因而UNIX BSD就发明了socket这种东西,socket屏蔽了各个协议的通讯细节,使得程序员无需关注协议自己,直接使用socket提供的接口来进行互联的不一样主机间的进程的通讯。这就比如操做系统给咱们提供了使用底层硬件功能的系统调用,经过系统调用咱们能够方便的使用磁盘(文件操做),使用内存,而无需本身去进行磁盘读写,内存管理。socket其实也是同样的东西,就是提供了tcp/ip协议的抽象,对外提供了一套接口,同过这个接口就能够统1、方便的使用tcp/ip协议的功能了。

  其实站在你的角度上看,socket就是一个模块。咱们经过调用模块中已经实现的方法创建两个进程之间的链接和通讯。也有人将socket说成ip+port,由于ip是用来标识互联网中的一台主机的位置,而port是用来标识这台机器上的一个应用程序。 因此咱们只要确立了ip和port就能找到一个应用程序,而且使用socket模块来与之通讯。

 三.套接字socket的发展史及分类

   基于文件类型的套接字家族

  套接字家族的名字:AF_UNIX

  基于网络类型的套接字家族

  套接字家族的名字:AF_INET

 四.基于TCP和UDP两个协议下socket的通信流程

   1.TCP和UDP对比

    TCP(Transmission Control Protocol)可靠的、面向链接的协议(eg:打电话)、传输效率低全双工通讯(发送缓存&接收缓存)、面向字节流。使用TCP的应用:Web浏览器;文件传输程序。

    UDP(User Datagram Protocol)不可靠的、无链接的服务,传输效率高(发送前时延小),一对1、一对多、多对1、多对多、面向报文(数据包),尽最大努力服务,无拥塞控制。使用UDP的应用:域名系统 (DNS);视频流;IP语音(VoIP)。

    直接看图对比其中差别

    继续往下看

    TCP和UDP下socket差别对比图:

   2.TCP协议下的socket

    基于TCP的socket通信流程图片:

    虽然上图将通信流程中的大体描述了一下socket各个方法的做用,可是仍是要总结一下通信流程(下面一段内容)

    先从服务器端提及。服务器端先初始化Socket,而后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待客户端链接。在这时若是有个客户端初始化一个Socket,而后链接服务器(connect),若是链接成功,这时客户端与服务器端的链接就创建了。客户端发送数据请求,服务器端接收请求并处理请求,而后把回应数据发送给客户端,客户端读取数据,最后关闭链接,一次交互结束

    上代码感觉一下,须要建立两个文件,文件名称随便起,为了方便看,个人两个文件名称为tcp_server.py(服务端)和tcp_client.py(客户端),将下面的server端的代码拷贝到tcp_server.py文件中,将下面client端的代码拷贝到tcp_client.py的文件中,而后先运行tcp_server.py文件中的代码,再运行tcp_client.py文件中的代码,而后在pycharm下面的输出窗口看一下效果。

    server端代码示例(若是比喻成打电话)

1
2
3
4
5
6
7
8
9
10
import  socket
sk  =  socket.socket()
sk.bind(( '127.0.0.1' , 8898 ))   #把地址绑定到套接字
sk.listen()           #监听连接
conn,addr  =  sk.accept()  #接受客户端连接
ret  =  conn.recv( 1024 )   #接收客户端信息
print (ret)        #打印客户端信息
conn.send(b 'hi' )         #向客户端发送信息
conn.close()        #关闭客户端套接字
sk.close()         #关闭服务器套接字(可选)

    client端代码示例

1
2
3
4
5
6
7
import  socket
sk  =  socket.socket()            # 建立客户套接字
sk.connect(( '127.0.0.1' , 8898 ))     # 尝试链接服务器
sk.send(b 'hello!' )
ret  =  sk.recv( 1024 )          # 对话(发送/接收)
print (ret)
sk.close()             # 关闭客户套接字

    socket绑定IP和端口时可能出现下面的问题:

     解决办法:

1
2
3
4
5
6
7
8
9
10
11
12
13
#加入一条socket配置,重用ip和端口
import  socket
from  socket  import  SOL_SOCKET,SO_REUSEADDR
sk  =  socket.socket()
sk.setsockopt(SOL_SOCKET,SO_REUSEADDR, 1 #在bind前加,容许地址重用
sk.bind(( '127.0.0.1' , 8898 ))   #把地址绑定到套接字
sk.listen()           #监听连接
conn,addr  =  sk.accept()  #接受客户端连接
ret  =  conn.recv( 1024 )    #接收客户端信息
print (ret)               #打印客户端信息
conn.send(b 'hi' )         #向客户端发送信息
conn.close()        #关闭客户端套接字
sk.close()         #关闭服务器套接字(可选)

    可是若是你加上了上面的代码以后仍是出现这个问题:OSError: [WinError 10013] 以一种访问权限不容许的方式作了一个访问套接字的尝试。那么只能换端口了,由于你的电脑不支持端口重用。

    记住一点,用socket进行通讯,必须是一收一发对应好。

  提一下:网络相关或者须要和电脑上其余程序通讯的程序才须要开一个端口。

  

  在看UDP协议下的socket以前,咱们还须要加一些内容来说:看代码

    server端

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
     import  socket
from  socket  import  SOL_SOCKET,SO_REUSEADDR
sk  =  socket.socket()
# sk.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)
sk.bind(( '127.0.0.1' , 8090 ))
sk.listen()
conn,addr  =  sk.accept()   #在这阻塞,等待客户端过来链接
while  True :
     ret  =  conn.recv( 1024 )   #接收消息  在这仍是要阻塞,等待收消息
     ret  =  ret.decode( 'utf-8' )   #字节类型转换为字符串中文
     print (ret)
     if  ret  = =  'bye' :         #若是接到的消息为bye,退出
         break
     msg  =  input ( '服务端>>' )   #服务端发消息
     conn.send(msg.encode( 'utf-8' ))
     if  msg  = =  'bye' :
         break
 
conn.close()
sk.close()

    client端

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
     import  socket
sk  =  socket.socket()
sk.connect(( '127.0.0.1' , 8090 ))  #链接服务端
 
while  True :
     msg  =  input ( '客户端>>>' )   #input阻塞,等待输入内容
     sk.send(msg.encode( 'utf-8' ))
     if  msg  = =  'bye' :
         break
     ret  =  sk.recv( 1024 )
     ret  =  ret.decode( 'utf-8' )
     print (ret)
     if  ret  = =  'bye' :
         break
sk.close()

  你会发现,第一个链接的客户端能够和服务端收发消息,可是第二个链接的客户端发消息服务端是收不到的

    缘由解释:
      tcp属于长链接,长链接就是一直占用着这个连接,这个链接的端口被占用了,第二个客户端过来链接的时候,他是能够链接的,可是处于一个占线的状态,就只能等着去跟服务端创建链接,除非一个客户端断开了(优雅的断开能够,若是是强制断开就会报错,由于服务端的程序还在第一个循环里面),而后就能够进行和服务端的通讯了。什么是优雅的断开呢?看代码。
    server端代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
     import  socket
from  socket  import  SOL_SOCKET,SO_REUSEADDR
sk  =  socket.socket()
# sk.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #容许地址重用,这个东西都说能解决问题,我很是不建议你们这么作,容易出问题
sk.bind(( '127.0.0.1' , 8090 ))
sk.listen()
# 第二步演示,再加一层while循环
while  True :     #下面的代码所有缩进进去,也就是循环创建链接,可是无论怎么聊,只能和一个聊,也就是另一个优雅的断了以后才能和另一个聊
                 #它不能同时和好多人聊,仍是长链接的缘由,一直占用着这个端口的链接,udp是能够的,而后咱们学习udp
     conn,addr  =  sk.accept()   #在这阻塞,等待客户端过来链接
     while  True :
         ret  =  conn.recv( 1024 )   #接收消息  在这仍是要阻塞,等待收消息
         ret  =  ret.decode( 'utf-8' )   #字节类型转换为字符串中文
         print (ret)
         if  ret  = =  'bye' :         #若是接到的消息为bye,退出
             break
         msg  =  input ( '服务端>>' )   #服务端发消息
         conn.send(msg.encode( 'utf-8' ))
         if  msg  = =  'bye' :
             break
     conn.close()

    client端代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
     import  socket
sk  =  socket.socket()
sk.connect(( '127.0.0.1' , 8090 ))  #链接服务端
 
while  True :
     msg  =  input ( '客户端>>>' )   #input阻塞,等待输入内容
     sk.send(msg.encode( 'utf-8' ))
     if  msg  = =  'bye' :
         break
     ret  =  sk.recv( 1024 )
     ret  =  ret.decode( 'utf-8' )
     print (ret)
     if  ret  = =  'bye' :
         break
# sk.close()

  强制断开链接以后的报错信息:

    

   3.UDP协议下的socket

    老样子!先上图!

    基于UDP的socket通信流程:

     总结一下UDP下的socket通信流程

      先从服务器端提及。服务器端先初始化Socket,而后与端口绑定(bind),recvform接收消息,这个消息有两项,消息内容和对方客户端的地址,而后回复消息时也要带着你收到的这个客户端的地址,发送回去,最后关闭链接,一次交互结束

      上代码感觉一下,须要建立两个文件,文件名称随便起,为了方便看,个人两个文件名称为udp_server.py(服务端)和udp_client.py(客户端),将下面的server端的代码拷贝到udp_server.py文件中,将下面cliet端的代码拷贝到udp_client.py的文件中,而后先运行udp_server.py文件中的代码,再运行udp_client.py文件中的代码,而后在pycharm下面的输出窗口看一下效果。

      server端代码示例

1
2
3
4
5
6
7
import  socket
udp_sk  =  socket.socket( type = socket.SOCK_DGRAM)    #建立一个服务器的套接字
udp_sk.bind(( '127.0.0.1' , 9000 ))         #绑定服务器套接字
msg,addr  =  udp_sk.recvfrom( 1024 )
print (msg)
udp_sk.sendto(b 'hi' ,addr)                  # 对话(接收与发送)
udp_sk.close()                          # 关闭服务器套接字

      client端代码示例

1
2
3
4
5
6
import  socket
ip_port = ( '127.0.0.1' , 9000 )
udp_sk = socket.socket( type = socket.SOCK_DGRAM)
udp_sk.sendto(b 'hello' ,ip_port)
back_msg,addr = udp_sk.recvfrom( 1024 )
print (back_msg.decode( 'utf-8' ),addr)

 五.粘包现象

  说粘包以前,咱们先说两个内容,1.缓冲区、2.windows下cmd窗口调用系统指令

   5.1  缓冲区(下面粘包现象的图里面还有关于缓冲区的解释)
    

  5.2 windows下cmd窗口调用系统指令(linux下没有写出来,你们仿照windows的去摸索一下吧)

    a.首先ctrl+r,弹出左下角的下图,输入cmd指令,肯定
      

    b.在打开的cmd窗口中输入dir(dir:查看当前文件夹下的全部文件和文件夹),你会看到下面的输出结果。

      

      另外还有ipconfig(查看当前电脑的网络信息),在windows没有ls这个指令(ls在linux下是查看当前文件夹下全部文件和文件夹的指令,和windows下的dir是相似的),那么没有这个指令就会报下面这个错误

      

   5.3 粘包现象(两种)

    先上图:(本图是我作出来为了让小白同窗有个大体的了解用的,其中不少地方更加的复杂,那就须要未来你们有多余的精力的时候去作一些深刻的研究了,这里我就不带你们搞啦)

    

     MTU简单解释:

MTU是Maximum Transmission Unit的缩写。意思是网络上传送的最大数据包。MTU的单位是字节。 
大部分网络设备的MTU都是1500个字节,也就是1500B。若是本机一次须要发送的数据比网关的MTU大,
大的数据包就会被拆开来传送,这样会产生不少数据包碎片,增长丢包率,下降网络速度

    关于上图中提到的Nagle算法等建议你们去看一看Nagle算法、延迟ACK、linux下的TCP_NODELAY和TCP_CORK,这些内容等大家把python学好之后再去研究吧,网络的内容实在太多啦,也就是说你们须要努力的过程还很长,加油!

  超出缓冲区大小会报下面的错误,或者udp协议的时候,你的一个数据包的大小超过了你一次recv能接受的大小,也会报下面的错误,tcp不会,可是超出缓存区大小的时候,确定会报这个错误。

   

  5.4 模拟一个粘包现象

    在模拟粘包以前,咱们先学习一个模块subprocess。
1
2
3
4
5
6
7
8
9
10
import  subprocess
cmd  =  input ( '请输入指令>>>' )
res  =  subprocess.Popen(
     cmd,                      #字符串指令:'dir','ipconfig',等等
     shell = True ,               #使用shell,就至关于使用cmd窗口
     stderr = subprocess.PIPE,   #标准错误输出,凡是输入错误指令,错误指令输出的报错信息就会被它拿到
     stdout = subprocess.PIPE,   #标准输出,正确指令的输出结果被它拿到
)
print (res.stdout.read().decode( 'gbk' ))
print (res.stderr.read().decode( 'gbk' ))

      注意:

        若是是windows,那么res.stdout.read()读出的就是GBK编码的,在接收端需要用GBK解码

        且只能从管道里读一次结果,PIPE称为管道。

     下面是subprocess和windows上cmd下的指令的对应示意图:subprocess的stdout.read()和stderr.read(),拿到的结果是bytes类型,因此须要转换为字符串打印出来看。
    

    

    好,既然咱们会使用subprocess了,那么咱们就经过它来模拟一个粘包

    tcp粘包演示(一):

      先从上面粘包现象中的第一种开始: 接收方没有及时接收缓冲区的包,形成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候仍是从缓冲区拿上次遗留的数据,产生粘包) 
      server端代码示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
cket  import  *
import  subprocess
 
ip_port = ( '127.0.0.1' , 8080 )
BUFSIZE = 1024
 
tcp_socket_server = socket(AF_INET,SOCK_STREAM)
tcp_socket_server.setsockopt(SOL_SOCKET,SO_REUSEADDR, 1 )
tcp_socket_server.bind(ip_port)
tcp_socket_server.listen( 5 )
 
while  True :
     conn,addr = tcp_socket_server.accept()
     print ( '客户端>>>' ,addr)
 
     while  True :
         cmd = conn.recv(BUFSIZE)
         if  len (cmd)  = =  0 : break
 
         res = subprocess.Popen(cmd.decode( 'gbk' ),shell = True ,
                          stdout = subprocess.PIPE,
                          stdin = subprocess.PIPE,
                          stderr = subprocess.PIPE)
 
         stderr = res.stderr.read()
         stdout = res.stdout.read()
         conn.send(stderr)
         conn.send(stdout)

      client端代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
import  socket
ip_port  =  ( '127.0.0.1' , 8080 )
size  =  1024
tcp_sk  =  socket.socket(socket.AF_INET,socket.SOCK_STREAM)
res  =  tcp_sk.connect(ip_port)
while  True :
     msg = input ( '>>: ' ).strip()
     if  len (msg)  = =  0 : continue
     if  msg  = =  'quit' : break
 
     tcp_sk.send(msg.encode( 'utf-8' ))
     act_res = tcp_sk.recv(size)
     print ( '接收的返回结果长度为>' , len (act_res))
     print ( 'std>>>' ,act_res.decode( 'gbk' ))  #windows返回的内容须要用gbk来解码,由于windows系统的默认编码为gbk

      tcp粘包演示(二):发送数据时间间隔很短,数据也很小,会合到一块儿,产生粘包

      server端代码示例:(若是两次发送有必定的时间间隔,那么就不会出现这种粘包状况,试着在两次发送的中间加一个time.sleep(1))

1
2
3
4
5
6
7
8
9
10
11
12
13
14
from  socket  import  *
ip_port = ( '127.0.0.1' , 8080 )
 
tcp_socket_server = socket(AF_INET,SOCK_STREAM)
tcp_socket_server.bind(ip_port)
tcp_socket_server.listen( 5 )
conn,addr = tcp_socket_server.accept()
data1 = conn.recv( 10 )
data2 = conn.recv( 10 )
 
print ( '----->' ,data1.decode( 'utf-8' ))
print ( '----->' ,data2.decode( 'utf-8' ))
 
conn.close()

      client端代码示例:

1
2
3
4
5
6
7
8
import  socket
BUFSIZE = 1024
ip_port = ( '127.0.0.1' , 8080 )
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
# res=s.connect_ex(ip_port)
res = s.connect(ip_port)
s.send( 'hi' .encode( 'utf-8' ))
s.send( 'meinv' .encode( 'utf-8' ))

      示例二的结果:所有被第一个recv接收了

    

 

     udp粘包演示:注意:udp是面向包的,因此udp是不存在粘包的
      server端代码示例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
import  socket
from  socket  import  SOL_SOCKET,SO_REUSEADDR,SO_SNDBUF,SO_RCVBUF
sk  =  socket.socket( type = socket.SOCK_DGRAM)
# sk.setsockopt(SOL_SOCKET,SO_RCVBUF,80*1024)
sk.bind(( '127.0.0.1' , 8090 ))
msg,addr  =  sk.recvfrom( 1024 )
while  True :
     cmd  =  input ( '>>>>' )
     if  cmd  = =  'q' :
         break
     sk.sendto(cmd.encode( 'utf-8' ),addr)
     msg,addr  =  sk.recvfrom( 1032 )
     # print('>>>>', sk.getsockopt(SOL_SOCKET, SO_SNDBUF))
     # print('>>>>', sk.getsockopt(SOL_SOCKET, SO_RCVBUF))
     print ( len (msg))
     print (msg.decode( 'utf-8' ))
 
sk.close()

       client端代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
import  socket
from  socket  import  SOL_SOCKET,SO_REUSEADDR,SO_SNDBUF,SO_RCVBUF
sk  =  socket.socket( type = socket.SOCK_DGRAM)
# sk.setsockopt(SOL_SOCKET,SO_RCVBUF,80*1024)
sk.bind(( '127.0.0.1' , 8090 ))
msg,addr  =  sk.recvfrom( 1024 )
while  True :
     cmd  =  input ( '>>>>' )
     if  cmd  = =  'q' :
         break
     sk.sendto(cmd.encode( 'utf-8' ),addr)
     msg,addr  =  sk.recvfrom( 1024 )
     # msg,addr = sk.recvfrom(1218)
     # print('>>>>', sk.getsockopt(SOL_SOCKET, SO_SNDBUF))
     # print('>>>>', sk.getsockopt(SOL_SOCKET, SO_RCVBUF))
     print ( len (msg))
     print (msg.decode( 'utf-8' ))
 
sk.close()

    在udp的代码中,咱们在server端接收返回消息的时候,咱们设置的recvfrom(1024),那么当我输入的执行指令为‘dir’的时候,dir在我当前文件夹下输出的内容大于1024,而后就报错了,报的错误也是下面这个:

  

    解释缘由:是由于udp是面向报文的,意思就是每一个消息是一个包,你接收端设置接收大小的时候,必需要比你发的这个包要大,否则一次接收不了就会报这个错误,而tcp不会报错,这也是为何ucp会丢包的缘由之一,这个和咱们上面缓冲区那个错误的报错缘由是不同的。  

  补充两个问题:

1
2
3
4
5
6
7
8
9
10
11
12
补充问题一:为什么tcp是可靠传输,udp是不可靠传输
 
     tcp在数据传输时,发送端先把数据发送到本身的缓存中,而后协议控制将缓存中的数据发往对端,对端返回一个ack = 1 ,发送端则清理缓存中的数据,对端返回ack = 0 ,则从新发送数据,因此tcp是可靠的。
     而udp发送数据,对端是不会返回确认信息的,所以不可靠
 
补充问题二:send(字节流)和sendall
 
     send的字节流是先放入己端缓存,而后由协议控制将缓存内容发往对端,若是待发送的字节流大小大于缓存剩余空间,那么数据丢失,用sendall就会循环调用send,数据不会丢失,通常的小数据就用send,由于小数据也用sendall的话有些影响代码性能,简单来说就是还多 while 循环这个代码呢。
  
用UDP协议发送时,用sendto函数最大能发送数据的长度为: 65535 -  IP头( 20 ) – UDP头( 8 )= 65507 字节。用sendto函数发送数据时,若是发送数据长度大于该值,则函数会返回错误。(丢弃这个包,不进行发送)
 
用TCP协议发送时,因为TCP是数据流协议,所以不存在包大小的限制(暂不考虑缓冲区的大小),这是指在用send函数时,数据长度参数不受限制。而实际上,所指定的这段数据并不必定会一次性发送出去,若是这段数据比较长,会被分段发送,若是比较短,可能会等待和下一次数据一块儿发送。

  粘包的缘由:主要仍是由于接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所形成的

六.粘包的解决方案

  解决方案(一):

     问题的根源在于,接收端不知道发送端将要传送的字节流的长度,因此解决粘包的方法就是围绕,如何让发送端在发送数据前,把本身将要发送的字节流总大小让接收端知晓,而后接收端发一个确认消息给发送端,而后发送端再发送过来后面的真实内容,接收端再来一个死循环接收完全部数据。
     

    看代码示例:

      server端代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
import  socket,subprocess
ip_port = ( '127.0.0.1' , 8080 )
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
s.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR,  1 )
 
s.bind(ip_port)
s.listen( 5 )
 
while  True :
     conn,addr = s.accept()
     print ( '客户端' ,addr)
     while  True :
         msg = conn.recv( 1024 )
         if  not  msg: break
         res = subprocess.Popen(msg.decode( 'utf-8' ),shell = True ,\
                             stdin = subprocess.PIPE,\
                          stderr = subprocess.PIPE,\
                          stdout = subprocess.PIPE)
         err = res.stderr.read()
         if  err:
             ret = err
         else :
             ret = res.stdout.read()
         data_length = len (ret)
         conn.send( str (data_length).encode( 'utf-8' ))
         data = conn.recv( 1024 ).decode( 'utf-8' )
         if  data  = =  'recv_ready' :
             conn.sendall(ret)
     conn.close()

      client端代码示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import  socket,time
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
res = s.connect_ex(( '127.0.0.1' , 8080 ))
 
while  True :
     msg = input ( '>>: ' ).strip()
     if  len (msg)  = =  0 : continue
     if  msg  = =  'quit' : break
 
     s.send(msg.encode( 'utf-8' ))
     length = int (s.recv( 1024 ).decode( 'utf-8' ))
     s.send( 'recv_ready' .encode( 'utf-8' ))
     send_size = 0
     recv_size = 0
     data = b''
     while  recv_size < length:
         data + = s.recv( 1024 )
         recv_size + = len (data)
 
 
     print (data.decode( 'utf-8' ))

  解决方案(二):

    经过struck模块将须要发送的内容的长度进行打包,打包成一个4字节长度的数据发送到对端,对端只要取出前4个字节,而后对这四个字节的数据进行解包,拿到你要发送的内容的长度,而后经过这个长度来继续接收咱们实际要发送的内容。不是很好理解是吧?哈哈,不要紧,看下面的解释~~
       为何要说一下这个模块呢,由于解决方案(一)里面你发现,我每次要先发送一个个人内容的长度,须要接收端接收,并切须要接收端返回一个确认消息,我发送端才能发后面真实的内容,这样是为了保证数据可靠性,也就是接收双方能顺利沟通,可是多了一次发送接收的过程,为了减小这个过程,咱们就要使struck来发送你须要发送的数据的长度,来解决上面咱们所说的经过发送内容长度来 解决粘包的问题

    struck模块的使用:struct模块中最重要的两个函数是pack()打包, unpack()解包。

    pack():#我在这里只介绍一下'i'这个int类型

1
2
3
4
5
6
7
import  struct
a = 12
# 将a变为二进制
bytes = struct.pack( 'i' ,a)
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
struct.pack( 'i' , 1111111111111 ) 若是 int 类型数据太大会报错struck.error
struct.error:  'i'  format  requires  - 2147483648  < =  number < =  2147483647  #这个是范围

    unpack():

1
2
3
# 注意,unpack返回的是tuple !!
 
a, = struct.unpack( 'i' ,bytes)  #将bytes类型的数据解包后,拿到int类型数据

  好,到这里咱们将struck这个模块将int类型的数据打包成四个字节的方法了,那么咱们就来使用它解决粘包吧。

  先看一段伪代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import  json,struct
#假设经过客户端上传1T:1073741824000的文件a.txt
 
#为避免粘包,必须自定制报头
header = { 'file_size' : 1073741824000 , 'file_name' : '/a/b/c/d/e/a.txt' , 'md5' : '8f6fbf8347faa4924a76856701edb0f3' #1T数据,文件路径和md5值
 
#为了该报头能传送,须要序列化而且转为bytes,由于bytes只能将字符串类型的数据转换为bytes类型的,全部须要先序列化一下这个字典,字典不能直接转化为bytes
head_bytes = bytes(json.dumps(header),encoding = 'utf-8' #序列化并转成bytes,用于传输
 
#为了让客户端知道报头的长度,用struck将报头长度这个数字转成固定长度:4个字节
head_len_bytes = struct.pack( 'i' , len (head_bytes))  #这4个字节里只包含了一个数字,该数字是报头的长度
 
#客户端开始发送
conn.send(head_len_bytes)  #先发报头的长度,4个bytes
conn.send(head_bytes)  #再发报头的字节格式
conn.sendall(文件内容)  #而后发真实内容的字节格式
 
#服务端开始接收
head_len_bytes = s.recv( 4 #先收报头4个bytes,获得报头长度的字节格式
x = struct.unpack( 'i' ,head_len_bytes)[ 0 #提取报头的长度
 
head_bytes = s.recv(x)  #按照报头长度x,收取报头的bytes格式
header = json.loads(json.dumps(header))  #提取报头
 
#最后根据报头的内容提取真实的数据,好比
real_data_len = s.recv(header[ 'file_size' ])
s.recv(real_data_len)

  下面看正式的代码:

  server端代码示例:报头:就是消息的头部信息,咱们要发送的真实内容为报头后面的内容。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
import  socket,struct,json
import  subprocess
phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR, 1 #忘了这是干什么的了吧,地址重用?想起来了吗~
 
phone.bind(( '127.0.0.1' , 8080 ))
phone.listen( 5 )
while  True :
     conn,addr = phone.accept()
     while  True :
         cmd = conn.recv( 1024 )
         if  not  cmd: break
         print ( 'cmd: %s'  % cmd)
         res = subprocess.Popen(cmd.decode( 'utf-8' ),
                              shell = True ,
                              stdout = subprocess.PIPE,
                              stderr = subprocess.PIPE)
         err = res.stderr.read()
         if  err:
             back_msg = err
         else :
             back_msg = res.stdout.read()
         conn.send(struct.pack( 'i' , len (back_msg)))  #先发back_msg的长度
         conn.sendall(back_msg)  #在发真实的内容
         #其实就是连续的将长度和内容一块儿发出去,那么整个内容的前4个字节就是咱们打包的后面内容的长度,对吧
         
     conn.close()

  client端代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
import  socket,time,struct
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
res = s.connect_ex(( '127.0.0.1' , 8080 ))
while  True :
     msg = input ( '>>: ' ).strip()
     if  len (msg)  = =  0 : continue
     if  msg  = =  'quit' : break
     s.send(msg.encode( 'utf-8' ))   #发送给一个指令
     l = s.recv( 4 )      #先接收4个字节的数据,由于咱们将要发送过来的内容打包成了4个字节,因此先取出4个字节
     x = struct.unpack( 'i' ,l)[ 0 ]   #解包,是一个元祖,第一个元素就是咱们的内容的长度
     print ( type (x),x)
     # print(struct.unpack('I',l))
     r_s = 0
     data = b''
     while  r_s < x:     #根据内容的长度来继续接收4个字节后面的内容。
         r_d = s.recv( 1024 )
         data + = r_d
         r_s + = len (r_d)
     # print(data.decode('utf-8'))
     print (data.decode( 'gbk' ))  #windows默认gbk编码
相关文章
相关标签/搜索