https://www.cnblogs.com/zingp/p/6863170.html 

一:IO模型介绍

IO发生时涉及的对象和步骤

  对于一个网络IO(network IO),它会涉及到两个系统对象,一个是调用这个IO的process (or thread),另一个就是系统内核(kernel)

  当一个read操作发生时,该操作会经历两个阶段:

  1)等待数据准备 (Waiting for the data to be ready)

  2)将数据从内核拷贝到进程中(Copying the data from the kernel to the process)

  记住这两点很重要,因为这些IO模型的区别就是在两个阶段上各有不同的情况

 

  1 import socket
2 server=socket.socket()
3 server.bind(("localhost",6969))
4 server.listen()
5 print("等待用户链接")
6 while True:
7 conn,addr=server.accept()
8 while True:
9 conn.send(("%s have connected to server"%addr).encode())
10 data=conn.recv(1024)
11 print("from client",data.decode())
12
13
14 import socket
15 client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
16 client.connect(("localhost",6969))
17 while True:
18 data=client.recv(1024)
19 print("from server:",data.decode())
20 client.send("hellow".encode())

阻塞IO

 import socket,time
server=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
server.bind(("localhost",6969))
server.listen()
server.setblocking(False)#设置为非阻塞,默认为阻塞
print("等待用户链接")
while True:
try:
conn,addr=server.accept()
conn.send("you have connected to server".encode())
data=conn.recv(1024)
print("from client",data.decode())
conn.close()
except Exception as e:
print(e)
time.sleep(4)
##########
import socket,time
client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
while True:
client.connect(("localhost", 6969))
data=client.recv(1024)
print("from server:",data.decode())
client.send("hellow".encode())
time.sleep(2)
break

非阻塞IO

由于设置了非阻塞IO(setblocking())所以在accept()的时候会报错,因为抓住了错误,所以开始会输出错误信息 ,有个问题就是服务端接收不到了客户端的数据

IO multiplexing:包括select,epoll,有些地方也称这种IO方式为event driven IO

注意1:select函数返回结果中如果有文件可读了,那么进程就可以通过调用accept()或recv()来让kernel将位于内核中准备到的数据copy到用户区

一:流程

二:IO多路复用的触发方式

  • 水平触发
  • 边缘触发

三:select实例

 import socket
import select
server=socket.socket()
server.bind(("localhost",6969))
server.listen()
while True:
r,w,e=select.select([server,],[],[],5)
#rlist -- wait until ready for reading
#wlist -- wait until ready for writing
#xlist -- wait for an ``exceptional condition''
#阻塞等待链接的时间
for i in r:
print(r)
conn,addr=i.accept()
print(conn)
print(addr)
print("hellow")
print(">>>>>")
#################### import socket,time
client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
while True:
client.connect(("localhost", 6969))
print(client)
data=client.recv(1024)
print("from server:",data.decode())
client.send("hellow".encode())

select实现IO多路复用

在不调用accept的时候,会反复的输出hellow和>是因为select的触发方式为水平触发

 import select,socket
server=socket.socket()
server.bind(("localhost",6969))
server.listen(5)
inp=[server,]
while True:
r, w, e = select.select(inp, [], [])
for obj in r:
if obj == server:
print("r:", r)
print(len(r))
print(obj)
conn, addr = obj.accept()
inp.append(conn)
else:
print("obj:",obj)
data=obj.recv(1024).decode()
print(">>:",data)
data=input("回答%s:"%str(addr))
obj.send(data.encode())
###################################
import socket
client=socket.socket()
client.connect(("localhost",6969))
print(client)
while True:
inp=input(">>>>:")
client.send(inp.encode())
data=client.recv(1024).decode()
print(data)

select实现监听多链接

异步IO全程无阻塞

 import selectors
import socket
def accept(server,mask):
conn,addr=server.accept()
print("client_sock:%s\nclient_addr:%s"%(conn,addr))
sel.register(conn,selectors.EVENT_READ,read)#注册,将conn和read函数绑定
def read(conn,mask):
try:#window如果客户端断开链接会报错,但是如果是linux客户端断开会发空数据,检测断开手段不一样
data=conn.recv(1024)
print(data.decode())
conn.send(data)
except Exception as e:
print("close:%s"%conn)
sel.unregister(conn)#解除绑定
conn.close()
if __name__ == '__main__': sel=selectors.DefaultSelector()#生成一个selector的对象
print("sel:",sel)
server=socket.socket()
server.bind(("localhost",6969))
server.listen()
sel.register(server,selectors.EVENT_READ,accept)#注册,将server与accept绑定
while True:
events=sel.select()#相当于select.select()#检测是否有链接或已连接的socket是否发送数据
print("events:",events)
print("\n")
for key,mask in events:
print("key:",type(key),key)#(SelectorKey(fileobj=<socket.socket fd=536, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 6969)>, fd=536, events=1, data=<function accept at 0x02C14E40>)
callback=key.data
callback(key.fileobj,mask)#调用相应的函数

selctor实现IO多路复用

												

python第10天(下)的更多相关文章

  1. python基础——面向对象进阶下

    python基础--面向对象进阶下 1 __setitem__,__getitem,__delitem__ 把对象操作属性模拟成字典的格式 想对比__getattr__(), __setattr__( ...

  2. python python中那些双下划线开头的那些函数都是干啥用用的

    1.写在前面 今天遇到了__slots__,,所以我就想了解下python中那些双下划线开头的那些函数都是干啥用用的,翻到了下面这篇博客,看着很全面,我只了解其中的一部分,还不敢乱下定义. 其实如果足 ...

  3. windows 10 64bit下安装Tensorflow+Keras+VS2015+CUDA8.0 GPU加速

    原文地址:http://www.jianshu.com/p/c245d46d43f0 写在前面的话 2016年11月29日,Google Brain 工程师团队宣布在 TensorFlow 0.12 ...

  4. python实现指定目录下批量文件的单词计数:并发版本

    在 文章 <python实现指定目录下批量文件的单词计数:串行版本>中, 总体思路是: A. 一次性获取指定目录下的所有符合条件的文件 -> B. 一次性获取所有文件的所有文件行 - ...

  5. 如何在Windows下开发Python:在cmd下运行Python脚本+如何使用Python Shell(command line模式和GUI模式)+如何使用Python IDE

    http://www.crifan.com/how_to_do_python_development_under_windows_environment/ 本文目的 希望对于,如何在Windows下, ...

  6. python进阶10 MySQL补充 编码、别名、视图、数据库修改

    python进阶10 MySQL补充    编码.别名.视图.数据库修改 一.编码问题 #MySQL级别编码 #修改位置: /etc/mysql/mysql.conf.d/mysqld.cnf def ...

  7. tensor搭建--windows 10 64bit下安装Tensorflow+Keras+VS2015+CUDA8.0 GPU加速

    windows 10 64bit下安装Tensorflow+Keras+VS2015+CUDA8.0 GPU加速 原文见于:http://www.jianshu.com/p/c245d46d43f0 ...

  8. libjingler-0.6.2在windows和ubuntu 10.04下的编译(Google Talk)

    Libjingle版本:0.6.2 所需的资源:         gtest-1.6.0.zip         http://download.csdn.net/detail/cl_gamer/48 ...

  9. Win 10环境下6sV2.1模型编译心得

    最新版本6sV2.1模型是通过FORTRAN95编写的,2017年11月代码编写完成,2018年11月发布在模型官网上.通常我们在使用过程中都是调用模型的.exe可执行文件,而下载下来的是FORTRA ...

  10. Python 入门 之 双下方法

    Python 入门 之 双下方法 1.双下方法 ​ 定义:双下方法是特殊方法,它是解释器提供的 由双下划线加方法名加双下划线 方法名的具有特殊意义的方法,双下方法主要是python源码程序员使用的,我 ...

随机推荐

  1. https://oi-wiki.org/

    OI网站 https://oi-wiki.org/

  2. lesson03

    3.1. 画 点 3.2. 基准平面 (重要) 1. 关于 点 的使用() 1.画一条直线,在线上画一个点(利用该点占该线段的百分比画出),通过该点画一条直线 2. 画一个长方体,定位到上表面.选择( ...

  3. 监控MySQL|Redis|MongoDB的执行语句(go-sniffer)

    上节回顾:https://www.cnblogs.com/dotnetcrazy/p/9986873.html 以CentOS为例: 1.环境 PS:如果不需要Golang环境,可以编译后把执行文件c ...

  4. mongodb 3.6 集群搭建:分片+副本集

    mongodb是最常用的nosql数据库,在数据库排名中已经上升到了前六.这篇文章介绍如何搭建高可用的mongodb(分片+副本)集群. 在搭建集群之前,需要首先了解几个概念:路由,分片.副本集.配置 ...

  5. beego框架的最简单登入演示

    一.controllers逻辑代码 func (c *UserController) Get() { c.TplName="login.html" } func (c *UserC ...

  6. echarts地图详解

    $(function() { // 路径配置 require.config({ paths : { // echarts: 'http://echarts.baidu.com/build/dist' ...

  7. 如何重写Java中的equals方法

    Java中,只有8种基本类型不是对象,例如:4种整形类型(byte, short, int,long),2种浮点类型(flout, double),boolean, char不是对象,其他的所有类型, ...

  8. SMB(Server Message Block) Protocal Research

    catalogue . 什么是SMB . 基础SMB协议 . SMB Clients and Servers 1. 什么是SMB 全世界有很多的客户端和服务端软件实现了SMB协议,全部windows工 ...

  9. 老男孩Python全栈学习 S9 日常作业 003

    1.有变量name = "aleX leNb" 完成如下操作: # 移除 name 变量对应的值两边的空格,并输出处理结果 # 移除name变量左边的"al"并 ...

  10. crm 权限设计

    先在项目中创建  app rbac的models.py from django.db import models class Permission(models.Model): "" ...