zoukankan      html  css  js  c++  java
  • python 并发编程(socketserver)

    下面的例子是简单的ssh 登录,其实也就是客户端把指令发送给服务器。服务器把结果返还给客户端,客户端再在终端展现

    服务端代码: #Author:BigBao #Date:2018/7/18 # 我们之前没有实现并发的原因是,我们之前是链接循环加通信循环,我们只有建立连接后才能通信,通信的过程中腾不出手来建立连接 # 所以现在我们得把链接循环和通信循环给分开,一个class一直在建立连接,一个class一直在通信 # 我们之前的是我们必须创建过链接之后立马去通信,通过过程中部可以去建立连接 ''' socketserver 封装了多进程,多线程 同时还封装了多路复用,把我们的程序性能发挥到机制 ''' import socketserver import subprocess import json import struct # 通信循环 class MyTcpHandler(socketserver.BaseRequestHandler): # BaseRequestHandler 专门用来负责处理通信相关信息的 def handle(self): # 这里必须定义一个handle方法,而且方法名必须是handle, sockerserver 会自动去调用handle这个方法 print(self.request) # 这里的self.request 相当于我们之前看到的conn那个对象( conn,client_addr=server.accept() ) while True: try: recv_cliend_cmd = self.request.recv(1024) #接收客户端的指令 if not recv_cliend_cmd:break # 下面就要对客户端发送的指令进行处理了 obj=subprocess.Popen(recv_cliend_cmd.decode('utf-8'),shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) stdout = obj.stdout.read() stderr = obj.stderr.read() # 下面开始创建报头(随意写的),这里我们真正用到的就是字典里的total_size header_dic={ 'total_size':len(stdout)+len(stderr), 'filename':'xxx.mp4', 'md5sum':'8f6fbf8347faa4924a76856701edb0f3' } header_json = json.dumps(header_dic) header_bytes = header_json.encode('utf-8') self.request.send(struct.pack('i',len(header_bytes))) # 这个发送过去为固定的字节数 4 个,所以客户端第一次接收四个字节即可 self.request.send(header_bytes) self.request.send(stdout) self.request.send(stderr) except ConnectionResetError:break self.request.close() # 连接循环 if __name__ == '__main__': server = socketserver.ThreadingTCPServer(('127.0.0.1',8080),MyTcpHandler) #基于Tcp的多线程服务端 server.serve_forever() # 客户端发来请求,就建立一个连接,server 立马造一个线程,然后再把MyTcpHandler 类实例化得到一个对象,触发对象下面的handle方法,把连接丢给 handle 方法(handle方法下的self.request 就是建立的连接) # 也就是说客户端来一个连接,建立后就会触发通信循环的的handle 方法,我们可以看一下self.request 打印出来的东西就是本地IP:端口 客户端IP:端口
    客户端代码
    
    #Author:BigBao
    #Date:2018/7/18
    import socket
    import struct
    import json
    client = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    client.connect(('127.0.0.1',8080))
    while True:
        cmd = input('please input your cmd: ').strip()
        if not cmd:continue
        client.send(cmd.encode('utf-8'))
        obj = client.recv(4)
        header_size = struct.unpack('i',obj)[0]
        header_bytes = client.recv(header_size)
        header_json = header_bytes.decode('utf-8')
        header_dic = json.loads(header_json)
        total_size = header_dic['total_size']
        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()
    

     我们上面处理的黏包问题是发生在服务端发数据给客户端,客户端的数据接收不完全的解决方案。要是涉及到两边都有黏包问题的话,解决方案也是上面一样解决

  • 相关阅读:
    javaweb 安全传输签名机制
    GC详解及Minor GC和Full GC触发条件总结
    程序员的人性思考(上)
    LeetCode 845——数组中的最长山脉
    LeetCode 386——字典序的第 K 小数字
    LeetCode 386——字典序排数
    LeetCode 135——分发糖果
    Ubuntu 16.04 安装显卡驱动后循环登录和无法设置分辨率的一种解决方案
    在 Ubuntu 下安装 Deepin 的 QQ、微信、百度云和迅雷等软件
    腾讯 2019 春季暑期实习生提前批笔试——技术研究与数据分析
  • 原文地址:https://www.cnblogs.com/smail-bao/p/9336332.html
Copyright © 2011-2022 走看看