协议使用场景:
udp协议(User Datagram Protocol) 即时通讯类的软件 :qq 微信 飞信 歪歪
tcp协议(Transmission Control Protocol) 发邮件\文件的时候
TCP协议:
面向连接\可靠\慢\对传递的数据的长短没有要求
两台机器之间要想传递信息必须先建立连接
之后在有了连接的基础上,进行信息的传递
可靠 : 数据不会丢失 不会重复被接收
慢 : 每一次发送的数据还要等待结果
import socket sk = socket.socket() sk.bind(('127.0.0.1',9001)) sk.listen() while True: conn,addr = sk.accept() while True: msg = input('>>>') conn.send(msg.encode('utf-8')) if msg.upper() == 'Q': break content = conn.recv(1024).decode('utf-8') if content.upper() == 'Q': break print(content) conn.close() sk.close()
import socket sk = socket.socket() sk.connect(('127.0.0.1',9001)) while True: ret = sk.recv(1024).decode('utf-8') if ret.upper() == 'Q':break print(ret) msg = input('>>>') sk.send(msg.encode('utf-8')) if msg.upper() == 'Q': break sk.close()
当应用程序希望通过 TCP 与另一个应用程序通信时,它会发送一个通信请求。这个请求必须被送到一个确切的地址。
在双方“握手”之后,TCP 将在两个应用程序之间建立一个全双工 (full-duplex) 的通信。这个全双工的通信将占用两个计算机之间的通信线路,直到它被一方或双方关闭为止。
序列号seq:占4个字节,用来标记数据段的顺序,TCP把连接中发送的所有数据字节都编上一个序号,第一个字节的编号由本地随机产生;给字节编上序号后,就给每一个报文段指派一个序号;序列号seq就是这个报文段中的第一个字节的数据编号。
确认号ack:占4个字节,期待收到对方下一个报文段的第一个数据字节的序号;序列号表示报文段携带数据的第一个字节的编号;而确认号指的是期望接收到下一个字节的编号;因此当前报文段最后一个字节的编号+1即为确认号。
确认ACK:占1位,仅当ACK=1时,确认号字段才有效。ACK=0时,确认号无效
同步SYN:连接建立时用于同步序号。当SYN=1,ACK=0时表示:这是一个连接请求报文段。若同意连接,则在响应报文段中使得SYN=1,ACK=1。因此,SYN=1表示这是一个连接请求,或连接接受报文。SYN这个标志位只有在TCP建产连接时才会被置1,握手完成后SYN标志位被置0。
终止FIN:用来释放一个连接。FIN=1表示:此报文段的发送方的数据已经发送完毕,并要求释放运输连接
PS:ACK、SYN和FIN这些大写的单词表示标志位,其值要么是1,要么是0;ack、seq小写的单词表示序号。
UDP协议:
无连接\不可靠\快\不能传输过长的数据
机器之间传递信息不需要建立连接 直接发就行
不可靠 : 数据有可能丢失
import socket sk = socket.socket(type=socket.SOCK_DGRAM) sk.bind(('127.0.0.1',9001)) while True: msg,client_addr = sk.recvfrom(1024) print(msg.decode('utf-8')) content = input('>>>') sk.sendto(content.encode('utf-8'),client_addr) sk.close()
import socket sk = socket.socket(type=socket.SOCK_DGRAM) server_addr = ('127.0.0.1',9001) while True: content = input('>>>') if content.upper() == 'Q':break sk.sendto(content.encode('utf-8'),server_addr) msg = sk.recv(1024).decode('utf-8') if msg.upper() == 'Q':break print(msg) sk.close()