socketserver实现并发
程序员文章站
2022-07-05 14:38:52
TCP协议的socket一次只能和一个客户端通信, 而socketsever可以时间和多个客户端通信. socketserver是在socket的基础上进行了一层封装, 它底层还是调用的socket. 我们通过以下代码来看下socketserver如何使用: import socketserver ......
tcp协议的socket一次只能和一个客户端通信, 而socketsever可以时间和多个客户端通信.
socketserver是在socket的基础上进行了一层封装, 它底层还是调用的socket.
我们通过以下代码来看下socketserver如何使用:
import socketserver # 引入模块 class myserver(socketserver.baserequesthandler): # 定义一个类, 继承socketserver模块中的baserequesthandler类 def handle(self): # # 写一个handle方法, 定死的(约束). 必须叫这个方法 while 1: from_client_msg = self.request.recv(1024) # 接收消息 print(from_client_msg.decode('utf-8')) server_msg = input('服务端说>>>') self.request.send(server_msg.encode('utf-8')) # 发送消息 if __name__ == '__main__': ip_port = ('127.0.0.1', 8008) server = socketserver.threadingtcpserver(ip_port, myserver) # 将ip,端口和定义的类传进socketserver.threadingtcpserver这个类.实例化一个对象 server.serve_forever() # 使用创建的对象调用server.serve_forever()方法. 这个方法的作用是让服务一直开着
import socket client = socket.socket() client.connect(('127.0.0.1', 8008)) while 1: client_msg = input('请输入>>>') client.send(client_msg.encode('utf-8')) from_server_msg = client.recv(1024) print(from_server_msg.decode('utf-8')) if client_msg == "byebye": break client.close()
简单解释以下什么是threadingtcpserver,多线程?
多线程就是我们的服务端通过多条线程同时和多个客户端进行沟通, 每条线程都对应一个客户端.
其实基于tcp的套接字, 关键就是两个循环, 一个连接循环, 一个通信循环
socketserver模块中分为两大类: server类(解决连接问题) 和 request类(解决通信问题)
server类:
request类:
继承关系:
以上面的代码中的代码为例, 分析socketserver源码:
ip_port = ('127.0.0.1', 8008)
server = socketserver.threadingtcpserver(ip_port, myserver)
server.serve_forever()
查找属性的顺序: threadingtcpserver->threadingmixin->tcpserver->baseserver
- 实例化得到server,先找类threadingtcpserver的__init__,在tcpserver中找到,进而执行server_bind,server_active
- 找server下的serve_forever,在baseserver中找到,进而执行self._handle_request_noblock(),该方法同样是在baseserver中
- 执行self._handle_request_noblock()进而执行request, client_address = self.get_request()(就是tcpserver中的self.socket.accept()),然后执行self.process_request(request, client_address)
- 在threadingmixin中找到process_request,开启多线程应对并发,进而执行process_request_thread,执行self.finish_request(request, client_address)
- 上述四部分完成了链接循环,本部分开始进入处理通讯部分,在baseserver中找到finish_request,触发我们自己定义的类的实例化,去找__init__方法,而我们自己定义的类没有该方法,则去它的父类也就是baserequesthandler中找....
源码分析总结:
基于tcp的socketserver我们自己定义的类中的
- self.server即套接字对象
- self.request即一个链接
- self.client_address即客户端地址
基于udp的socketserver我们自己定义的类中的
1.self.request是一个元组(第一个元素是客户端发来的数据,第二部分是服务端的udp套接字对象).
2.self.client_address即客户端地址
上一篇: BZOJ1853: [Scoi2010]幸运数字(容斥原理)
下一篇: 爬虫,百度搜索热点排行
推荐阅读
-
socket的多线程实现
-
Linux 创建子进程执行任务的实现方法
-
Springboot项目redisTemplate实现轻量级消息队列
-
啰嗦的 java,简洁的 lombok —— lombok 的使用及简单实现单例模式注解
-
SpringBoot中并发定时任务的实现、动态定时任务的实现(看这一篇就够了)
-
Spring Boot Security OAuth2 实现支持JWT令牌的授权服务器
-
springboot2.0.3源码篇 - 自动配置的实现,发现也不是那么复杂
-
在ubuntu16.04上创建matlab的快捷方式(实现方法)
-
微信小程序 在线支付功能的实现
-
spring boot2.0.4集成druid,用jmeter并发测试工具调用接口,druid查看监控的结果