基于python
select.select模块通信的实例讲解

要理解select.select模块其实主要就是要理解它的参数, 以及其三个返回值。

select()方法接收并监控3个通信列表,
第一个是所有的输入的data,就是指外部发过来的数据,第2个是监控和接收所有要发出去的data(outgoing
data),第3个监控错误信息在网上一直在找这个select.select的参数解释, 但实在是没有,
哎...自己硬着头皮分析了一下。  
 

readable, writable, exceptional = select.select(inputs, outputs,
inputs)

第一个参数就是服务器端的socket, 第二个是我们在运行过程中存储的客户端的socket, 第三个存储错误信息。

重点是在返回值, 第一个返回的是可读的list, 第二个存储的是可写的list, 第三个存储的是错误信息的list。

这个也不必深究, 看看代码自己分析下就能有大概理解。

网上所有关于select.select的代码都是差不多的, 但是有些不能运行, 或是不全。我自己重新写了一份能运行的程序,
做了很多注释, 好好看看就能搞懂

服务器端:    

# coding: utf-8

import select

import socket

import Queue

from time import sleep

 

 

# Create a TCP/IP

server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

server.setblocking(False)

 

# Bind the socket to the port

server_address = ('localhost', 8090)

print ('starting up on %s port %s' % server_address)

server.bind(server_address)

 

# Listen for incoming connections

server.listen(5)

 

# Sockets from which we expect to read

inputs = [server]

 

# Sockets to which we expect to write

# 处理要发送的消息

outputs = []

 

# Outgoing message queues (socket: Queue)

message_queues = {}

 

while inputs:

  # Wait for at least one of the sockets to be
ready for processing

  print ('waiting for the next event')

  # 开始select 监听, 对input_list 中的服务器端server
进行监听

  # 一旦调用socket的send, recv函数,将会再次调用此模块

  readable, writable, exceptional =
select.select(inputs, outputs, inputs)

 

  # Handle inputs

  # 循环判断是否有客户端连接进来, 当有客户端连接进来时select 将触发

  for s in readable:

    #
判断当前触发的是不是服务端对象, 当触发的对象是服务端对象时,说明有新客户端连接进来了

    #
表示有新用户来连接

    if s is
server:

     
# A "readable" socket is ready to accept a connection

     
connection, client_address = s.accept()

     
print ('connection from', client_address)

     
# this is connection not server

     
connection.setblocking(0)

     
# 将客户端对象也加入到监听的列表中, 当客户端发送消息时 select 将触发

     
inputs.append(connection)

 

     
# Give the connection a queue for data we want to send

     
# 为连接的客户端单独创建一个消息队列,用来保存客户端发送的消息

     
message_queues[connection] = Queue.Queue()

    else:

     
# 有老用户发消息, 处理接受

     
# 由于客户端连接进来时服务端接收客户端连接请求,将客户端加入到了监听列表中(input_list),
客户端发送消息将触发

     
# 所以判断是否是客户端对象触发

     
data = s.recv(1024)

     
# 客户端未断开

     
if data != '':

       
# A readable client socket has data

       
print ('received "%s" from %s' % (data, s.getpeername()))

       
# 将收到的消息放入到相对应的socket客户端的消息队列中

       
message_queues[s].put(data)

       
# Add output channel for response

       
# 将需要进行回复操作socket放到output 列表中, 让select监听

       
if s not in outputs:

         
outputs.append(s)

     
else:

       
# 客户端断开了连接, 将客户端的监听从input列表中移除

       
# Interpret empty result as closed connection

       
print ('closing', client_address)

       
# Stop listening for input on the connection

       
if s in outputs:

         
outputs.remove(s)

       
inputs.remove(s)

       
s.close()

 

       
# Remove message queue

       
# 移除对应socket客户端对象的消息队列

       
del message_queues[s]

 

  # Handle outputs

  # 如果现在没有客户端请求, 也没有客户端发送消息时, 开始对发送消息列表进行处理,
是否需要发送消息

  # 存储哪个客户端发送过消息

  for s in writable:

    try:

     
# 如果消息队列中有消息,从消息队列中获取要发送的消息

     
message_queue = message_queues.get(s)

     
send_data = ''

     
if message_queue 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)



客户端:    

# coding: utf-8

import socket

 

 

messages = ['This is the message ', 'It will be sent ', 'in parts
', ]

 

server_address = ('localhost', 8090)

 

# Create aTCP/IP socket

 

socks = [socket.socket(socket.AF_INET, socket.SOCK_STREAM),
socket.socket(socket.AF_INET, socket.SOCK_STREAM), ]

 

# Connect thesocket to the port where the server is listening

 

print ('connecting to %s port %s' % server_address)

# 连接到服务器

for s in socks:

  s.connect(server_address)

 

for index, message in enumerate(messages):

  # Send messages on both sockets

  for s in socks:

    print ('%s:
sending "%s"' % (s.getsockname(), message str(index)))

   
s.send(bytes(message str(index)).decode('utf-8'))

  # Read responses on both sockets

 

for s in socks:

  data = s.recv(1024)

  print ('%s: received "%s"' % (s.getsockname(),
data))

  if data != "":

    print
('closingsocket', s.getsockname())

   
s.close()

写代码过程中遇到了两个问题, 一是如何判断客户端已经关闭了socket连接, 后来自己分析了下, 如果关闭了客户端socket,
那么此时服务器端接收到的data就是'', 加个这个判断。二是如果服务器端关闭了socket,
一旦在调用socket的相关方法都会报错, 不管socket是不是用不同的容器存储的(意思是说list_1存储了socket1,
list_2存储了socket1, 我关闭了socket1, 两者都不能在调用这个socket了)

服务器端:

客户端:

以上这篇基于python select.select模块通信的实例讲解就是小编分享给大家的全部内容了

基于pythonselect.select模块通信的实例讲解的更多相关文章

  1. python select.select模块通信全过程详解

    要理解select.select模块其实主要就是要理解它的参数, 以及其三个返回值.select()方法接收并监控3个通信列表, 第一个是所有的输入的data,就是指外部发过来的数据,第2个是监控和接 ...

  2. 实例讲解基于 React+Redux 的前端开发流程

    原文地址:https://segmentfault.com/a/1190000005356568 前言:在当下的前端界,react 和 redux 发展得如火如荼,react 在 github 的 s ...

  3. 基于tcpdump实例讲解TCP/IP协议

    前言 虽然网络编程的socket大家很多都会操作,但是很多还是不熟悉socket编程中,底层TCP/IP协议的交互过程,本文会一个简单的客户端程序和服务端程序的交互过程,使用tcpdump抓包,实例讲 ...

  4. Android单片机与蓝牙模块通信实例代码

    Android单片机与蓝牙模块通信实例代码 参考路径:http://www.jb51.net/article/83349.htm 啦啦毕业了,毕业前要写毕业设计,需要写一个简单的蓝牙APP进行交互,通 ...

  5. 基于Python中numpy数组的合并实例讲解

    基于Python中numpy数组的合并实例讲解 下面小编就为大家分享一篇基于Python中numpy数组的合并实例讲解,具有很好的参考价值,希望对大家有所帮助.一起跟随小编过来看看吧 Python中n ...

  6. 012 基于FPGA的网口通信实例设计【转载】

    一.网口通信设计分类 通过上面其他章节的介绍,网口千兆通信,可以使用TCP或者UDP协议,可以外挂PHY片或者不挂PHY片,总结下来就有下面几种方式完成通信: 图8‑17基于FPGA的网口通信实例设计 ...

  7. Python之基于socket和select模块实现IO多路复用

    '''IO指的是输入输出,一部分指的是文件操作,还有一部分网络传输操作,例如soekct就是其中之一:多路复用指的是利用一种机制,同时使用多个IO,例如同时监听多个文件句柄(socket对象一旦传送或 ...

  8. android 基于wifi模块通信开发

    这篇文章主要是我写完手机与wifi模块通信后所用来总结编写过程的文章,下面,我分几点来说一下编写的大概流程. 一.拉出按钮控件并设置它的点击事件 二.设置wifi权限 三.打开和关闭wifi 四.扫描 ...

  9. 【MySQL】分页查询实例讲解

    MySQL分页查询实例讲解 1. 前言 本文描述了团队在工作中遇到的一个MySQL分页查询问题,顺带讲解相关知识点,为后来者鉴.本文的重点不是"怎样"优化表结构和SQL语句,而是探 ...

随机推荐

  1. 前端学习笔记--CSS3

    本本记录了css3的样式:浏览器支持度.圆角边框.阴影.文字与文本.过渡.动画.2d旋转.3d旋转 浏览器支持度: 1.圆角边框 例:只要确定了x.y值,就能知道弧度 画一个圆形:长=宽,border ...

  2. MySQL远程连接不上的解决方法

    这篇文章主要为大家详细介绍了MySQL远程连接不上的解决方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下   之前遇到过MySQL本地可以连接但是远程连不上的问题,没有记录,今天在云上新申请的服 ...

  3. PL/SQL块与表达式

    一.块(Block) 是PL/SQL的基本执行单元,由定义部分,执行部分(必须)和例外处理部分组成. Declare /*定义部分――定义常量.变量.游标.例外.复杂数据类型*/ Begin /*执行 ...

  4. 通过trace分析优化器如何选择执行计划

    1. mysql> show variables like "optimizer_trace%"\G;*************************** 1. row * ...

  5. 2019牛客暑期多校训练营(第十场)Han Xin and His Troops——扩展中国剩余定理

    题意 求解 $n$ 个模方程 $x \equiv a (mod \ b)$,不保证模数互素($1 \leq n \leq 100$,$0 \leq b < a< 10^5$). 分析 套扩 ...

  6. sqlserver 删除表数据

    可以使用delete清空表delete from t表名 也可以使用truncate命令 truncate table 表名

  7. Laravel Model updating&updated 事件使用注意事项

    1 触发条件 1.1 updating 1.1.1 如果字段无变化,不会触发此事件. 1.1.2 除非更改至少一个字段的值 2 事件逻辑不会覆盖 2.1 Trait 中定义事件如下 /** * The ...

  8. exam9.3

    #   用  户  名  公园 计划 抽卡   总分  19 859乔屹 100 03:15:05 40 03:14:01   140 03:15:05 emm 怎么讲 T2我把自己优化掉了40分 优 ...

  9. tornado解析post数据的问题

    解析tornado查询参数: self.request.query_arguments self.get_query_argument[s](参数名称) 解析tornado的post参数: self. ...

  10. notepad++修改背景色