socket编程
什么是socket
Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。
所以,我们无需深入理解tcp/udp协议,socket已经为我们封装好了,我们只需要遵循socket的规定去编程,写出的程序自然就是遵循tcp/udp标准的。
基于文件类型的套接字家族
套接字家族的名字:AF_UNIX
unix一切皆文件,基于文件的套接字调用的就是底层的文件系统来取数据,两个套接字进程运行在同一机器,可以通过访问同一个文件系统间接完成通信
基于网络类型的套接字家族
套接字家族的名字:AF_INET
(还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现,所有地址家族中,AF_INET是使用最广泛的一个,python支持很多种地址家族,但是由于我们只关心网络编程,所以大部分时候我么只使用AF_INET)
套接字函数用法
服务端套接字函数
s.bind() 绑定(主机,端口号)到套接字
s.listen() 开始TCP监听
s.accept() 被动接受TCP客户的连接,(阻塞式)等待连接的到来
客户端套接字函数
s.connect() 主动初始化TCP服务器连接
s.connect_ex() connect()函数的扩展版本,出错时返回出错码,而不是抛出异常
公共用途的套接字函数
s.recv() 接收TCP数据
s.send() 发送TCP数据(send在待发送数据量大于己端缓存区剩余空间时,数据丢失,不会发完)
s.sendall() 发送完整的TCP数据(本质就是循环调用send,sendall在待发送数据量大于己端缓存区剩余空间时,数据不丢失,循环调用send直到发完)
s.recvfrom() 接收UDP数据
s.sendto() 发送UDP数据
s.getpeername() 连接到当前套接字的远端的地址
s.getsockname() 当前套接字的地址
s.getsockopt() 返回指定套接字的参数
s.setsockopt() 设置指定套接字的参数
s.close() 关闭套接字
面向锁的套接字方法
s.setblocking() 设置套接字的阻塞与非阻塞模式
s.settimeout() 设置阻塞套接字操作的超时时间
s.gettimeout() 得到阻塞套接字操作的超时时间
面向文件的套接字的函数
s.fileno() 套接字的文件描述符
s.makefile() 创建一个与该套接字相关的文件
重启服务端可能出现的错误
#加入一条socket配置,重用ip和端口 phone=socket(AF_INET,SOCK_STREAM) phone.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加 phone.bind(('127.0.0.1',8080))
基于tcp的套接字模拟打电话
服务端
import socket # 1.先买手机 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM) # socket.SOCK_STREAM指的是TCP协议 # 2.绑定电话卡 # phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1) # 就是它,在bind前加,防止操作系统未及时清理链接,导致端口被占用的问题 phone.bind(('127.0.0.1',8080)) # 端口范围0-65535,1-1024系统占用 # 3.开机 phone.listen(5) # 4.等电话 print('starting....') while True: # 链接循环 conn,addr = phone.accept() # (conn,client_addr) # print('=========>') # print(conn,addr) # 5.收发消息 while True: # 通信循环 try: # print('========>ready recv') data = conn.recv(1024) # 1024最大接收的字节数 conn.send(data.upper()) except ConnectionResetError: break # 6.挂电话 conn.close() # 7.关机 phone.close()
客户端
import socket # 1.先买手机 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM) # socket.SOCK_STREAM指的是TCP协议 # 2.打电话 phone.connect(('127.0.0.1',8080)) # 3.发收消息 while True: msg = input('>>:').strip() if not msg:continue phone.send(msg.encode('utf-8')) data = phone.recv(1024) print(data.decode('utf-8')) # 4.关闭 phone.close()
远程执行命令
服务端
from socket import * import subprocess server = socket(AF_INET,SOCK_STREAM) server.bind(('127.0.0.1',8080)) server.listen(5) while True: conn,addr = server.accept() while True: try: cmd = conn.recv(1024) if not cmd:break # linux系统中客户端断开不会抛异常,而是会不停收空 # 执行命令 cmd = cmd.decode('utf-8') # 调用模块,执行命令,并且收集命令的执行结果,而不是打印 obj = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, ) stdout = obj.stdout.read() stderr = obj.stderr.read() # 发送命令的结果 conn.send(stdout+stderr) except ConnectionResetError: break conn.close() server.close()
客户端
from socket import * client = socket(AF_INET,SOCK_STREAM) client.connect(('127.0.0.1',8080)) while True: cmd = input('>>:').strip() if not cmd:continue client.send(cmd.encode('utf-8')) data = client.recv(1024) print(data.decode('gbk')) client.close()
粘包问题
TCP(transport control protocol,传输控制协议)是面向连接的,面向流的,提供高可靠性服务。收发两端(客户端和服务器端)都要有一一成对的socket,因此,发送端为了将多个发往接收端的包,更有效的发到对方,使用了优化方法(Nagle算法),将多次间隔较小且数据量小的数据,合并成一个大的数据块,然后进行封包。这样,接收端,就难于分辨出来了,必须提供科学的拆包机制。 即面向流的通信是无消息保护边界的
两种情况下会产生粘包
1.发送端需要等缓冲区满才发送出去,造成粘包(发送数据时间间隔很短,数据了很小,会合到一起,产生粘包)
2.接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包)
远程执行命令粘包基本解决方法
服务端
from socket import * import subprocess import struct server = socket(AF_INET,SOCK_STREAM) server.bind(('127.0.0.1',8080)) server.listen(5) while True: conn,addr = server.accept() while True: try: cmd = conn.recv(1024) if not cmd:break # linux系统中客户端断开不会抛异常,而是会不停收空 # 执行命令 cmd = cmd.decode('utf-8') # 调用模块,执行命令,并且收集命令的执行结果,而不是打印 obj = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, ) stdout = obj.stdout.read() stderr = obj.stderr.read() # 第一步:制作报头 total_size = len(stdout) + len(stderr) header = struct.pack('i',total_size) # 第二步:先发报头(固定长度) conn.send(header) # 第三步:发送命令的结果 conn.send(stdout) conn.send(stderr) except ConnectionResetError: break conn.close() server.close()
客户端
from socket import * import struct client = socket(AF_INET,SOCK_STREAM) client.connect(('127.0.0.1',8080)) while True: cmd = input('>>:').strip() if not cmd:continue client.send(cmd.encode('utf-8')) # 第一步:收到报头:长度 header = client.recv(4) total_size = struct.unpack('i',header)[0] # 第二部:收完整真实的数据 recv_size = 0 res = b'' while recv_size < total_size: recv_data = client.recv(1024) res += recv_data recv_size += len(recv_data) print(recv_data.decode('gbk')) client.close()
解决粘包问题终极版
服务端
import subprocess import struct import json from socket import * server=socket(AF_INET,SOCK_STREAM) server.bind(('127.0.0.1',8086)) # print(server) server.listen(5) while True: conn,addr = server.accept() # print(conn) print(addr) while True: try: cmd = conn.recv(8096) if not cmd:break #针对linux #执行命令 cmd = cmd.decode('utf-8') #调用模块,执行命令,并且收集命令的执行结果,而不是打印 obj = subprocess.Popen(cmd, shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, ) stdout = obj.stdout.read() stderr = obj.stderr.read() # 1:先制作报头,报头里放:数据大小, md5, 文件 header_dic = { 'total_size':len(stdout)+len(stderr), 'md5': 'xxxxxxxxxxxxxxxxxxx', 'filename': 'xxxxx', 'xxxxx':'123123' } header_json = json.dumps(header_dic) header_bytes = header_json.encode('utf-8') header_size = struct.pack('i', len(header_bytes)) # 2: 先发报头的长度 conn.send(header_size) # 3:先发报头 conn.send(header_bytes) # 4:再发送真实数据 conn.send(stdout) conn.send(stderr) except ConnectionResetError: break conn.close() server.close()
客户端
import struct import json from socket import * client = socket(AF_INET,SOCK_STREAM) client.connect(('127.0.0.1',8086)) while True: cmd = input('>>: ').strip() if not cmd:continue client.send(cmd.encode('utf-8')) # 1:先收报头长度 obj = client.recv(4) header_size = struct.unpack('i', obj)[0] # 2:先收报头,解出报头内容 header_bytes = client.recv(header_size) header_json = header_bytes.decode('utf-8') header_dic = json.loads(header_json) print(header_dic) total_size = header_dic['total_size'] # 3:循环收完整数据 recv_size = 0 res = b'' while recv_size < total_size: recv_data = client.recv(1024) res += recv_data recv_size += len(recv_data) print(res.decode('gbk')) client.close()
UDP
udp是无链接的,先启动哪一端都不会报错
服务端
from socket import * server = socket(AF_INET,SOCK_DGRAM) server.bind(('127.0.0.1',8080)) while True: data,client_addr = server.recvfrom(1024) server.sendto(data.upper(),client_addr)
客户端
from socket import * client = socket(AF_INET,SOCK_DGRAM) while True: msg = input('>>:').strip() client.sendto(msg.encode('utf-8'),('127.0.0.1',8080)) data,server_addr = client.recvfrom(1024) print(data.decode('utf-8'))