select监听服务端
# can_read, can_write, _ = select.select(inputs, outputs, None, None)
#
# 第一个参数是我们需要监听可读的套接字, 第二个参数是我们需要监听可写的套接字, 第三个参数使我们需要监听异常的套接字, 第四个则是时间限制设置.
#
# 如果监听的套接字满足了可读可写条件, 那么所返回的can,read 或是 can_write就会有值了, 然后我们就可以利用这些返回值进行随后的操作了。相比较unix 的select模型, 其select函数的返回值是一个整型, 用以判断是否执行成功.
#
# 第一个参数就是服务器端的socket, 第二个是我们在运行过程中存储的客户端的socket, 第三个存储错误信息。
# 重点是在返回值, 第一个返回的是可读的list, 第二个存储的是可写的list, 第三个存储的是错误信息的
# list。
import select, socket, queue
from time import sleep
# 创造一个 TCP/IP连接
server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 设置socket为异步模式。
server.setblocking(False)
# 绑定IP地址和端口
# 创建一个IP,端口,本地IP
server_address = ('localhost', 8090)
print('starting up on %s port %s' % server_address)
# 绑定IP和端口
server.bind(server_address)
# 设置服务端监听数量
server.listen(3)
inputs = [server]
# 处理要发送的数据
outputs = []
# 输出的消息队列
message_queues = {}
while inputs:
# 等待至少一个连接开始调用线程
print("waiting for the next event")
# 开始select监听,对input_list中的服务器端的server开始进行监听
# 一旦调用socket的send,recv函数,将会再次调用此模块
# 这里监控三个参数,第一个返回的是可读的list, 第二个存储的是可写的list, 第三个存储的是错误信息的
readable, writable, exceptional = select.select(inputs, outputs, inputs)
# 处理输入
# 循环判断是否有客户端连接进来,当有客户端连接进来的时候select将进行触发
for s in readable:
# 判断当前触发的是不是服务端对象,当触发的是服务端对象, 说明有新的客户端连接进来了
if s is server:
# 和客户端建立连接
connection, client_address = s.accept()
# 打印客户端的链接
print('connection from', client_address)
# 设置socket为异步模式。
connection.setblocking(0)
# 将客户端对象也加入到监听列表中,当客户端发送消息的时候select也会触发。
inputs.append(connection)
# 为连接的客户端单独建立一个消息队列,用来保存客户端发送的消息
message_queues[connection] = queue.Queue()
else:
# 有老用户发送消息,处理接收
# 由于客户端连接进来的时候服务端接收客户端的请求,将客户端也加入到了监听列表。
# 客户端发送的消息将触发。
# 判断触发对象是否为客户端
# 接收客户端的信息
data = s.recv(1024)
# 客户端没有断开
if data != '':
# 一个可用的客户端发送了数据
print('received "%s" from %s' % (data, s.getpeername()))
# 将接受到的消息放到对应的客户端的消息队列中
message_queues[s].put(data)
# 将需要进行回复操作的socket放到output列表中,让select监听
# 这里我们如果收到消息将服务端添加进入输出列表。
# 然后select就会监听到,
if s not in outputs:
outputs.append(s)
else:
# 客户端被监听到有变化,不是有客户端传过来的消息,就是客户端断开了。
# 客户端断开了连接,将客户端的监听从input列表中移除
print('closing', client_address)
if s in outputs:
outputs.remove(s)
inputs.remove(s)
s.close()
# 移除对应的socket客户端对象的消息队列
del message_queues[s]
# 处理输出。
# 如果现在没有客户端的请求,也没有客户端发送消息时,开始对发送的消息列表进行处理。
# 判断是否需要发送消息。
# 存储哪个客户端发送了消息。
for s in writable:
try:
# 如果消息队列中有消息,就从消息队列中获取要发送的消息。
message_queue = message_queues.get(s)
sent_data = ''
if message_queues is not None:
# 如果队列为空的话不等(阻塞)
send_data = message_queue.get_nowait()
else:
# 客户端断开连接了
print('has closed')
# 如果队列为空的话,就代表着发送的消息完了,就需要从队列中删除。
except queue.Empty:
print('%s' % (s.getpeername()))
outputs.remove(s)
else:
# print "sending %s to %s " % (send_data, s.getpeername)
# print "send something"
if message_queue is not None:
s.send(send_data)
else:
print("has closed ")
# del message_queues[s]
# writable.remove(s)
# print "Client %s disconnected" % (client_address)
# # Handle "exceptional conditions"
# 处理异常的情况
# 如果有异常情况,那么就从输出队列中删除,然后关闭连接。
for s in exceptional:
print('exception condition on', s.getpeername())
# Stop listening for input on the connection
inputs.remove(s)
if s in outputs:
outputs.remove(s)
s.close()
# Remove message queue
del message_queues[s]
sleep(1)
select监听服务端的更多相关文章
- DSAPI多功能组件编程应用-HTTP监听服务端与客户端_指令版
前面介绍了DSAPI多功能组件编程应用-HTTP监听服务端与客户端的内容,这里介绍一个适用于更高效更快速的基于HTTP监听的服务端.客户端. 在本篇,你将见到前所未有的超简化超傻瓜式的HTTP监听服务 ...
- DSAPI HTTP监听服务端与客户端
本文中,演示了使用DSAPI.网络相关.HTTP监听,快速建立服务端和客户端. HTTP监听服务端的作用,是监听指定计算机端口,以实现与IIS相同的解析服务,提供客户端的网页请求,当然,这不仅仅是应用 ...
- DSAPI HTTP监听服务端与客户端_指令版
前面介绍了DSAPI多功能组件编程应用-HTTP监听服务端与客户端的内容,这里介绍一个适用于更高效更快速的基于HTTP监听的服务端.客户端. 在本篇,你将见到前所未有的超简化超傻瓜式的HTTP监听服务 ...
- DSAPI多功能组件编程应用-HTTP监听服务端与客户端
本文中,演示了使用DSAPI.网络相关.HTTP监听,快速建立服务端和客户端. HTTP监听服务端的作用,是监听指定计算机端口,以实现与IIS相同的解析服务,提供客户端的网页请求,当然,这不仅仅是应用 ...
- oracle监听服务开启
输入命令netca即可开启oracle的监听服务 弹出对话框 选择监听服务配置,单击下一步 选择增加监听,单击下一步 监听的名字,默认即可,下一步 监听链接的协议,默认TCP协议即可,下一步 监听默认 ...
- oracle 监听服务配置
最近在red hat 6.6虚拟机上安装了Oracle 11gR2数据库,安装完毕,使用没有问题,通过主机也可以访问到虚拟机上的数据库.然而,在重新启动虚拟机后,主机无法访问到数据库,提示错误: PS ...
- Oracle监听服务启动失败案例
在ORACLE测试服务器上还原恢复了一个数据库后,启动监听服务时出现了TNS-12541, TNS-12560,TNS-00511之类的错误,具体情况如下所示: [oracle@getlnx01 ad ...
- oracle数据库没有监听服务与实例服务(OracleServicesXX)的解决方法
不知道为什么,可能是因为更新系统的原因,过了一段时间,想打开oracle服务,发现居然没有任何oracle有关的服务了,但以前的数据库文件什么的都在,心想肯定是可以复原的,应该只是注册表的问题罢了.在 ...
- 服务器重启后Oracle监听服务没有自动启动的解决方案
最近一直在被这样一个问题烦恼,就是服务器断电重启后,Oracle监听服务没有正常自动启动(监听服务已经设置为自启动). 具体是这样的,监听服务设置为开机自启动,Oracle数据库服务设置为开机延时启动 ...
随机推荐
- RabbitMQ的简单封装
一般在工作中,都是直接使用已经封装好的mq的程序集进行功能开发.所以很多时候都没有去了解rabbitmq到底是如何封装(实现使用的).所以心血来潮,简单记录下自己对rabbitmq的简单封装 整体的思 ...
- mysql千万级大数据SQL查询优化30条经验(Mysql索引优化注意)
1.对查询进行优化,应尽量避免全表扫描,首先应考虑在 where 及 order by 涉及的列上建立索引. 2.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索 ...
- CORS与Django
前言 在前后端分离项目中,如何解决跨域请求是一个必须要面对的问题.因为前端和后端的数据交互会被浏览器的同源策略所挟持,在很早之前我在博客园发了一篇文章,大概就说了一下如何简单粗暴的解决跨域. 其实那种 ...
- JavaScript变量污染
定义过多的全局变量,有可能造成全局变量冲突,这种现象称为变量污染. 全局变量在全局作用域内外都是可见的.若是已经声明了一个全局变量,再以相同的关键字和标识符重新声明全局变量,后者的赋值会替代前者的赋值 ...
- 【JVM第二篇--类加载机制】类加载器与双亲委派模型
写在前面的话:本文是在观看尚硅谷JVM教程后,整理的学习笔记.其观看地址如下:尚硅谷2020最新版宋红康JVM教程 一.什么是类加载器 在类加载过程中,加载阶段有一个动作是"通过一个类的全限 ...
- socket里面那个又爱又恨的锁
查一个问题:结果看了一下软中断以及系统 所耗cpu,心中满是伤痕啊------- perf 结果一眼可以看到:主要是锁 那么这个lock 是用来干什么的呢?? A:TCP socket的使用者有两种: ...
- mysql之分区表
1.分区表概述: 1.分区表的主要意义在于,对于表结构进行划分,不同的数据进入不同的分区中,以便于在查询过程中,只查找指定分区的数据,减少数据库扫描的数据量. 2.虽然从逻辑上看分区表是一张表,但是底 ...
- 删除osd的正确方式
在ceph的集群当中关于节点的替换的问题,一直按照以前的方式进行的处理,处理的步骤如下: 停止osd进程 /etc/init.d/ceph stop osd.0 这一步是停止osd的进程,让其他的os ...
- 验证rbd的缓存是否开启
简单快速的在客户端验证rbd的cache是否开启 首先修改配置文件 在ceph.conf中添加: [client] rbd cache = true rbd cache writethrough un ...
- Python_PyQt5_eric6 做省市县筛选框
eric是PyQt5的图形化编辑工具,界面如下(另存为-桌面 查看大图) 下面是用eric6制作的 省市县 三级联动筛选框 (效果图+源码) 1 # -*- coding: utf-8 -*- 2 ...