day08 多线程socket 编程,tcp粘包处理
复习下socket 编程的步骤:

# 获取客户端ip地址和端口 开启一个连接实例动态导入模块:
实践:
1 lib下的aa
def test():
print("hehe")
class C(object):
name = "yang"
def __init__(self):
self.name = 'shenyang' lib外的test
import importlib
dd = importlib.import_module('lib.aa')
dd.test()
print(dd.C().name)

运行:


判断错误

和 if 判断 一样 但是更简洁,一句话就OK
继续socket 编程:

#!/usr/bin/env python3
# Author: Shen Yang
import socket,os
ip_address = '192.168.16.10'
port = 8888
bind_address = (ip_address,port)
server = socket.socket()
server.bind(bind_address)
server.listen()
while True:
conn,addr = server.accept()
while True:
data = conn.recv(1024).decode()
if not data:
print("丢失连接")
break
print("这是来自",addr,data)
cmd_res = os.popen(data).read()
conn.send(str(len(cmd_res.encode('utf-8'))).encode('utf-8'))
conn.send(cmd_res.encode('utf-8'))
server.close()
客户端
#!/usr/bin/env python3
# Author: Shen Yang
import socket
ip_address = '192.168.16.10'
# ip_address = '192.168.81.133'
port = 8888
conn_address = (ip_address,port)
client = socket.socket()
client.connect(conn_address)
while True:
cmd = input(":> ").encode('utf-8')
if len(cmd) == 0:
continue
client.send(cmd)
cmd_size = int(client.recv(1024).decode())
print(cmd_size)
recov_size = 0
recov_data = b''
while recov_size < cmd_size:
data = client.recv(1024)
#print(data)
recov_size += len(data)
#print(recov_size)
recov_data += data
else:
print("收完了,大小",recov_size)
print(recov_data.decode())
client.close()

上面的代码 在linux 发送 会出现粘包 客户端收到的是两个数据

客户端加发送过程:

#!/usr/bin/env python3
import socket,os
ip_address = '192.168.81.133'
port = 8888
bind_address = (ip_address,port)
server = socket.socket()
server.bind(bind_address)
server.listen()
while True:
conn,addr = server.accept()
while True:
data = conn.recv(1024).decode()
if not data:
print("丢失连接")
break
print("这是来自",addr,data)
cmd_res = os.popen(data).read()
conn.send(str(len(cmd_res.encode('utf-8'))).encode('utf-8'))
ack = conn.recv(1024).decode()
print(ack)
conn.send(cmd_res.encode('utf-8'))
server.close()

ssh_client
#!/usr/bin/env python3
import socket
ip_address = '192.168.81.133'
port = 8888
conn_address = (ip_address,port)
client = socket.socket()
client.connect(conn_address)
while True:
cmd = input(":> ").encode('utf-8')
if len(cmd) == 0:
continue
client.send(cmd)
cmd_size = int(client.recv(1024).decode())
print(cmd_size)
client.send("收到大小".encode('utf-8'))
recov_size = 0
recov_data = b''
while recov_size < cmd_size:
data = client.recv(1024)
#print(data)
recov_size += len(data)
#print(recov_size)
recov_data += data
else:
print("收完了,大小",recov_size)
print(recov_data.decode())
client.close()


使用 os.stat 获取文件详细信息,发送给客户端
另外一个解决粘包的方法:
只收指定大小的数据 永远不可能粘包

#!/usr/bin/env python3
import socket,os,hashlib
ip_address = '192.168.81.133'
port = 8888
bind_address = (ip_address,port)
server = socket.socket()
server.bind(bind_address)
server.listen()
while True:
conn,addr = server.accept()
while True:
print("等待新链接")
data = conn.recv(1024)
if not data:
print("丢失连接")
break
print("这是来自",addr,data)
cmd,file_name = data.decode().split()
if os.path.isfile(file_name):
m = hashlib.md5()
f = open(file_name,'rb')
file_size = os.stat(file_name).st_size
conn.send(str(file_size).encode())
ack = conn.recv(1024).decode()
print("确认:",ack)
for line in f:
m.update(line)
print("sending...")
conn.send(line)
f.close()
conn.send(m.hexdigest().encode())
print("发送完毕")
server.close()

ftp_client.py
#!/usr/bin/env python3
import socket,hashlib
ip_address = '192.168.81.133'
port = 8888
conn_address = (ip_address,port)
client = socket.socket()
client.connect(conn_address)
while True:
cmd = input("input your cmd:> ").strip()
if len(cmd) == 0:
continue
if cmd.startswith("get"):
client.send(cmd.encode('utf-8'))
file_name = cmd.split()[1]
server_response = client.recv(1024)
print("收到回应:",server_response)
client.send("收到".encode('utf-8'))
total_file_size = int(server_response.decode())
print("总的",total_file_size)
f = open(file_name + '.new','wb')
recov_file_size = 0
m = hashlib.md5()
while recov_file_size < total_file_size:
if total_file_size - recov_file_size > 1024:
size = 1024
else:
size = total_file_size - recov_file_size
print("最后收",size)
#print("收数据...")
data = client.recv(size)
m.update(data)
recov_file_size += len(data)
f.write(data)
else:
print("收完了,大小",recov_file_size,total_file_size)
f.close()
print("服务器端发送的原MD5",client.recv(1024).decode())
print("自己算出来的最终MD5",m.hexdigest())
client.close()

继承关系:

创建socket server 步骤:

下面是重写的handle:


import socketserver
class MyTCPHandler(socketserver.BaseRequestHandler):
def handle(self):
while True:
try:
self.data = self.request.recv(1024).strip()
print("来自",self.client_address)
print(self.data)
self.request.send(self.data.upper())
except ConnectionResetError as e:
print("客户端退出",e)
break
if __name__ == "__main__":
HOST,PORT="localhost",9999
server = socketserver.TCPServer((HOST,PORT),MyTCPHandler)
server.serve_forever()

client:
import socket
client = socket.socket() #声明socket类型,同时生成socket连接对象
#client.connect(('192.168.16.200',9999))
client.connect(('localhost',9999))
while True:
msg = input(">>:").strip()
if len(msg) == 0:continue
client.send(msg.encode("utf-8"))
data = client.recv(10240)
print("recv:",data.decode())
client.close()

2 使用多进程实现: (windos 下不行,不支持,但是linux 绝得好使)

2 处理单个请求,一般也用不到

3 一直收请求,直到收到一个明确的shutdown (每0.5秒检测是否给我发了shutdown 的信号)

4 告诉 server_forever() 关闭

5 地址的重用,不需要等待tcp断开


day08 多线程socket 编程,tcp粘包处理的更多相关文章
- socket编程 TCP 粘包和半包 的问题及解决办法
一般在socket处理大数据量传输的时候会产生粘包和半包问题,有的时候tcp为了提高效率会缓冲N个包后再一起发出去,这个与缓存和网络有关系. 粘包 为x.5个包 半包 为0.5个包 由于网络原因 一次 ...
- Socket编程--TCP粘包问题
TCP是个流协议,它存在粘包问题 产生粘包的原因是: TCP所传输的报文段有MSS的限制,如果套接字缓冲区的大小大于MSS,也会导致消息的分割发送. 由于链路层最大发送单元MTU,在IP层会进行数据的 ...
- python/socket编程之粘包
python/socket编程之粘包 粘包 只有TCP有粘包现象,UDP永远不会粘包. 首先需要掌握一个socket收发消息的原理 发送端可以是1k,1k的发送数据而接受端的应用程序可以2k,2k的提 ...
- Python全栈-网络编程-TCP粘包
一.什么是TCP粘包 C/S架构下,接收方不知道每个消息的发送间隙.也不知道每次应该提取多少个字节的数据,与此同时,TCP是面向连接的,面向流的,收发两端都要有,因此发送端为了将多个发往接收端的数据包 ...
- socket编程解决粘包和丢包问题
##socket 丢包粘包解决方式 采用固定头部长度(一般为4个字节),包头保存的是包体的长度 header+body 包头+包体 下面的例子不是按照上图中规定的格式编写的,但是思路都是一样的,先读出 ...
- Socket编程 Tcp和粘包
大多数程序员都要接触网络编程,Web开发天天和http打交道.稍微底层一点的程序员,就是TCP/UDP . 对程序员来说,Tcp/udp的核心是Socket编程. 我的浅薄的观点---------理解 ...
- Socket编程(4)TCP粘包问题及解决方案
① TCP是个流协议,它存在粘包问题 TCP是一个基于字节流的传输服务,"流"意味着TCP所传输的数据是没有边界的.这不同于UDP提供基于消息的传输服务,其传输的数据是有边界的.T ...
- Socket编程实践(5) --TCP粘包问题与解决
TCP粘包问题 由于TCP协议是基于字节流且无边界的传输协议, 因此很有可能产生粘包问题, 问题描述如下 对于Host A 发送的M1与M2两个各10K的数据块, Host B 接收数据的方式不确定, ...
- 查漏补缺:socket编程:TCP粘包问题和常用解决方案(上)
1.TCP粘包问题的产生(发送端) 由于TCP协议是基于字节流并且无边界的传输协议,因此很容易产生粘包问题.TCP的粘包可能发生在发送端,也可能发生在接收端.发送端的粘包是TCP协议本身引起的,TCP ...
随机推荐
- Date 对象 时间格式注意事项
Date 对象,是操作日期和时间的对象. Date 为内置的构造函数, 通过 new Date () 来获取当前本地日期与时间 const time = new Date console.log(ti ...
- 【css】css2实现两列三列布局的方法
前言 对于 flex 弹性布局相信大家都有所了解,它是 css3 中的属性,然而它具有一定的兼容性问题.楼主前几天面试时遇到了面试官需要设计一个两列布局,我当然就说父元素 flex 吧哩吧啦,然而需要 ...
- C# 对接腾讯企业邮接口----get/post请求
在无所知之的情况下.来了一个对接接口的任务,没办法,只能根据前端时候的经验硬着头皮上了,随后又整理了一下写的方法,主要包括了部门的创建.更新.删除.查找.然后他们的前提是token的获取 首先HTTP ...
- Android开发学习——小细节注意
Android中通过Intent调用其他应用的方法(转) Android中两种序列化方式的比较Serializable和Parcelable http://www.jcodecraeer.com/a/ ...
- Jquery 日期差函数 修改 对火狐进行兼容
function DateDiff(sDate1, sDate2) { //sDate1和sDate2是yyyy-MM-dd格式 var aDate, oDate1, oDate2, iDays; a ...
- python+selenium之断言Assertion
一.断言方法 断言是对自动化测试异常情况的判断. # -*- coding: utf-8 -*- from selenium import webdriver import unittest impo ...
- HDU 1007 Quoit Design最近点对( 分治法)
题意: 给出平面上的n个点,问任意点对之间的最短距离是多少? 思路: 先将所有点按照x坐标排序,用二分法将n个点一分为二个部分,递归下去直到剩下两或一个点.对于一个部分,左右部分的答案分别都知道,那么 ...
- 为Oracle Clusterware修改公用及私有网络接口
出于种种原因我们可能需要为已安装的Oracle集群软件修改其使用的公用或私有网络所使用的网络接口(How to Change Interconnect/Public Interface IP or S ...
- 【UML】时序图Sequence diagram(交互图)(转)
前言 UML时序图是UML动态图之一,它是强调时间顺序的交互图. 定义 时序图是显示按时间顺序排列的对象之间交互的图. 组成元素 对象 包括三种命名 ...
- Android(java)学习笔记109:Java中输入和输出流概念
程序在内存中运行,文件在磁盘上,把文件从磁盘上读入内存中来,当然是输入流了, 反之,把内存中的数据写到磁盘上的文件里面去就是输出.通常都是这样的,用了过滤流的情况复杂一些,则另当别论.