基于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. clean()方法的简单应用

    clean()方法主要用于验证相互依赖的字段,例如注册时,填写的“密码”和“确认密码”要相等时才符合要求. 在调用表单clean() 方法的时候,所有字段的验证方法已经执行完(表单字段的默认验证(如C ...

  2. TCP服务端实现并发

    socket 在 tcp 协议下通信 客户端 import socket ​ # 创建客户端TCP协议通信 c = socket.socket() # 与指定服务端握手 c.connect(('127 ...

  3. 一些VMware vCenter Appliance的默认用户名和密码

    一些VMware vCenter Appliance的默认用户名和密码 2014-03-30 17:30:03 flowershade_21 阅读数 13367更多 分类专栏: vmware   VM ...

  4. Vuex框架原理与源码分析

    Vuex是一个专为Vue服务,用于管理页面数据状态.提供统一数据操作的生态系统.它集中于MVC模式中的Model层,规定所有的数据操作必须通过 action - mutation - state ch ...

  5. luogu P3709 大爷的字符串题

    二次联通门 : luogu P3709 大爷的字符串题 /* luogu P3709 大爷的字符串题 莫队 看了半天题目 + 题解 才弄懂了要求什么... 维护两个数组 一个记录数字i出现了几次 一个 ...

  6. AGC037C Numbers on a Circle【构造】

    从后往前做,每次将\(B_i\)减去相邻两个数,注意如果最大的数没有变成初始状态,那么肯定要减,否则相邻两边的就减不了,所以用堆维护.根据辗转相除的复杂度,\(O(n\log^2 n)\). #inc ...

  7. redis配置数据持久化---APPEND ONLY MODE

    Redis配置数据持久化---APPEND ONLY MODE 2016年04月01日 19:05:11 阅读数:9918 Redis可以实现数据的持久化存储,即将数据保存到磁盘上. Redis的持久 ...

  8. 数据结构实验之图论五:从起始点到目标点的最短步数(BFS)

    分析:有向图里面找最短路径,原理就是每一步都走距离自己最近的路, 一旦发现走一步可以到,那么这个一定是最短的. #include <bits/stdc++.h> using namespa ...

  9. linux经常用的命令

    常用 安装包   centos   yum    /   Ubuntu  Debian  apt-get clear :清空终端       [cmd 下是cls] vi/vim  编辑器    详情 ...

  10. YII框架入门

    一.安装 php运行环境安装包:http://www.phpstudy.net/a.php/207.html Yii安装包: https://github.com/yiisoft/yii2/relea ...