socketserver 源码剖析:

socketserver 源码剖析【有图有真相】:

(一)、Socketserver 内部流程调用图:

  

 

  详解:

1、self.RequestHandlerClass() = MyClass() 转换为 执行这个方法 class MyClass(socketserer.BaseRquestHandler)。

2、 myclass 没有构造方法 __init__( ),从socketserer.BaseRquestHandler 父类 开始找,有构造函数 __init__( ),并且执行了一个 self.handle()方法 。

但是,还得从myclass() 类中开始找,即:class MyClass(socketserver.BaseRequestHandler):

def handle(self): # self = obj创建的实例化。 



################# 精简版的 socketserver 源码剖析 ##################!/usr/bin/env python# 详见老师分析:s13期。socketserver 源码分析剖析(一)和(二)。

import socketserver

class MyClass(socketserver.BaseRequestHandler):

def handle(self): #self = obj创建的实例化。        pass

obj = socketserver.ThreadingTCPServer((‘127.0.0.1‘, 9999), MyClass)obj.serve_forever()

# 原理剖析:窥一斑见全豹    # 创建socket对象    # accept --    # server_address = (‘127.0.0.1‘, 9999)    # RequestHandlerClass = MyClass  == ()    # self.RequestHandlerClass() = MyClass()  == 执行自己的handle()    # 1、obj封装了 self.RequestHandlerClass = MyClass    # 2、创建了socket,bind,listen

# 1、首先执行构造方法 __init__(),当所有的构造方法执行完了,意味着 实例化完成了,即 obj = socketserver.ThreadingTCPServer()# 2、遇到执行方法self.函数名(),然后在从头开始找执行。详见:s13 day10 socketserver() 源码分析二。


 2、ThreadingTCPServer源码剖析.

学会看源码非常重要!不能仅仅光会用!大赞~ 知道他的过程和实现~ 怎么学会看源码呢?多看然后画类图,如上图!!!

在理解的时候可以把他们想象为,把所有需要用的方法,都抓到ThreadingTCPServer

内部调用流程为:

  • 启动服务端程序.
  • 执行 TCPServer.__init__ 方法,创建服务端Socket对象并绑定 IP 和 端口
  • 执行 BaseServer.__init__ 方法,将自定义的继承自SocketServer.BaseRequestHandler 的类 MyRequestHandle赋值给 self.RequestHandlerClass
  • 执行 BaseServer.server_forever 方法,While 循环一直监听是否有客户端请求到达 ...
  • 当客户端连接到达服务器
  • 执行 ThreadingMixIn.process_request 方法,创建一个 “线程” 用来处理请求
  • 执行 ThreadingMixIn.process_request_thread 方法
  • 执行 BaseServer.finish_request 方法,执行 self.RequestHandlerClass()
  • 即:执行 自定义 MyRequestHandler 的构造方法(自动调用基类BaseRequestHandler的构造方法,在该构造方法中又会调用 MyRequestHandler的handle方法)

精简源码:

模拟Socekt Server的简化版本:


import socketimport threadingimport select

def process(request, client_address): #模拟定义的handle()方法,这个方法内的代码是socket server与Client端交互代码    print (request,client_address)    conn = request    conn.sendall(‘欢迎致电 10086,请输入1xxx,0转人工服务.‘)    flag = True    while flag:        data = conn.recv(1024)        if data == ‘exit‘:            flag = False        elif data == ‘0‘:            conn.sendall(‘通过可能会被录音.balabala一大推‘)        else:            conn.sendall(‘请重新输入.‘)

sk = socket.socket(socket.AF_INET, socket.SOCK_STREAM)sk.bind((‘127.0.0.1‘,8002))sk.listen(5)

while True:      #这里一个while循环循环监控sk文件句柄    r, w, e = select.select([sk,],[],[],1)    print (‘looping‘)    if sk in r: #当sk文件句柄发生变化的时候说明是新的客户端连接过来了        print (‘get request‘)        request, client_address = sk.accept()        t = threading.Thread(target=process, args=(request, client_address)) #创建一个线程,并调用自己定义的process方法执行~然后样客户端与之交互        t.daemon = False        t.start()

sk.close()

如精简代码可以看出,SocketServerThreadingTCPServer之所以可以同时处理请求得益于 select 和 Threading 两个东西,其实本质上就是在服务器端为每一个客户端创建一个线程,当前线程用来处理对应客户端的请求,所以,可以支持同时n个客户端链接(长连接)。


  • ForkingTCPServer 

       ForkingTCPServerThreadingTCPServer的使用和执行流程基本一致,只不过在内部分别为请求者建立 “线程”  和 “进程”。

#!/usr/bin/env python# -*- coding:utf-8 -*-import socketserver

class MyServer(socketserver.BaseRequestHandler):

def handle(self):        # print self.request,self.client_address,self.server        conn = self.request        conn.sendall(‘欢迎致电 10086,请输入1xxx,0转人工服务.‘)        Flag = True        while Flag:            data = conn.recv(1024)            if data == ‘exit‘:                Flag = False            elif data == ‘0‘:                conn.sendall(‘通过可能会被录音.balabala一大推‘)            else:                conn.sendall(‘请重新输入.‘)

if __name__ == ‘__main__‘:    server = socketserver.ForkingTCPServer((‘127.0.0.1‘,8009),MyServer)    server.serve_forever()

client

#!/usr/bin/env python# -*- coding:utf-8 -*-

import socket

ip_port = (‘127.0.0.1‘,8009)sk = socket.socket()sk.connect(ip_port)sk.settimeout(5)

while True:    data = sk.recv(1024)    print (‘receive:‘,data)    inp = input(‘please input:‘)    sk.sendall(inp)    if inp == ‘exit‘:        break

sk.close()

client

以上ForkingTCPServer 只是将 ThreadingTCPServer 实例中的代码:

server = SocketServer.ThreadingTCPServer((‘127.0.0.1‘,8009),MyRequestHandler)
变更为:
server = SocketServer.ForkingTCPServer((‘127.0.0.1‘,8009),MyRequestHandler)

来源: http://www.cnblogs.com/luotianshuai/p/5111587.html

来自为知笔记(Wiz)

时间: 2024-10-06 00:31:32

socketserver 源码剖析:的相关文章

socketserver源码剖析

这里选择的是python2.7(python3和2.7的源码基本类似) #!/usr/bin/env python # -*- coding:utf-8 -*- import SocketServer class MyServer(SocketServer.BaseRequestHandler): def handle(self): conn = self.request conn.sendall(bytes("欢迎致电老男孩", encoding="utf-8")

自动化运维Python系列之IO多路复用、SocketServer源码分析

IO多路复用 IO多路复用是指:通过一种机制,可以监视多个描述符,一旦某个系统描述符就绪(一般是读就绪或者写就绪)能够通知程序进行相应的读写操作 实例化例子就是在SocketServer模块中,客户端和服务端建立好连接,此时服务端通过监听conn这条链路,一旦客户端发送了数据,conn链路状态就发生变化,服务端就知道有数据要接收... Linux系统中同时存在select.pull.epoll三种IO多路复用机制 windows中只有select机制 1)select select本质上是通过设

Appuim源码剖析(Bootstrap)

Appuim源码剖析(Bootstrap) SkySeraph Jan. 26th 2017 Email:[email protected] 更多精彩请直接访问SkySeraph个人站点:www.skyseraph.com About Appuim Appium 是一个自动化测试开源工具,支持 iOS 平台和 Android 平台上的原生应用,web 应用和混合应用. 这里有很关键一点,跨平台.更多了解Appuim多平台支持相关信息,参考官方platform-support 相关概念 C/S 架

下载-深入浅出Netty源码剖析、Netty实战高性能分布式RPC、NIO+Netty5各种RPC架构实战演练三部曲视频教程

下载-深入浅出Netty源码剖析.Netty实战高性能分布式RPC.NIO+Netty5各种RPC架构实战演练三部曲视频教程 第一部分:入浅出Netty源码剖析 第二部分:Netty实战高性能分布式RPC 第三部分:NIO+Netty5各种RPC架构实战演练

Phaser实现源码剖析

在这里首先说明一下,由于Phaser在4.3代码里是存在,但并没有被开放出来供使用,但已经被本人大致研究了,因此也一并进行剖析. Phaser是一个可以重复利用的同步栅栏,功能上与CyclicBarrier和CountDownLatch相似,不过提供更加灵活的用法.也就是说,Phaser的同步模型与它们差不多.一般运用的场景是一组线程希望同时到达某个执行点后(先到达的会被阻塞),执行一个指定任务,然后这些线程才被唤醒继续执行其它任务. Phaser一般是定义一个parties数(parties一

【Java集合源码剖析】HashMap源码剖析

转载请注明出处:http://blog.csdn.net/ns_code/article/details/36034955 HashMap简介 HashMap是基于哈希表实现的,每一个元素是一个key-value对,其内部通过单链表解决冲突问题,容量不足(超过了阀值)时,同样会自动增长. HashMap是非线程安全的,只是用于单线程环境下,多线程环境下可以采用concurrent并发包下的concurrentHashMap. HashMap 实现了Serializable接口,因此它支持序列化,

转:【Java集合源码剖析】Vector源码剖析

转载请注明出处:http://blog.csdn.net/ns_code/article/details/35793865   Vector简介 Vector也是基于数组实现的,是一个动态数组,其容量能自动增长. Vector是JDK1.0引入了,它的很多实现方法都加入了同步语句,因此是线程安全的(其实也只是相对安全,有些时候还是要加入同步语句来保证线程的安全),可以用于多线程环境. Vector没有丝线Serializable接口,因此它不支持序列化,实现了Cloneable接口,能被克隆,实

下载BootStrap企业级应用培训课程(零基础、源码剖析,内部教材,项目实训)

全套500多课,附赠JS OOP编程,转一播放码.下载地址:http://pan.baidu.com/s/1kVLdZmf 第一季:基础篇,侧重于BootStrap 相关 API 详解.主要包含以下内容:Brackets前端开发工具详解.BootStrap框架三大核心-CSS.BootStrap框架三大核心-布局组件.BootStrap框架三大核心-JavaScript插件.附-BootStrap编码规范第二季:高级篇,侧重于BootStap源码解析与第三方扩展.主要包含以下内容:BootStr

菜鸟nginx源码剖析 框架篇(一) 从main函数看nginx启动流程(转)

俗话说的好,牵牛要牵牛鼻子 驾车顶牛,处理复杂的东西,只要抓住重点,才能理清脉络,不至于深陷其中,不能自拔.对复杂的nginx而言,main函数就是“牛之鼻”,只要能理清main函数,就一定能理解其中的奥秘,下面我们就一起来研究一下nginx的main函数. 1.nginx的main函数解读 nginx启动显然是由main函数驱动的,main函数在在core/nginx.c文件中,其源代码解析如下,涉及到的数据结构在本节仅指出其作用,将在第二节中详细解释. nginx main函数的流程图如下: