基于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. async/await中reject的问题

    promise 返回的 resolve 对象可能用 await 去接,但是 reject 无法用 await 接收到,所以要用 try catch 去处理 例如发送邮件的接口设置: async fun ...

  2. MOS管做开关之初理解

    杰杰 物联网IoT开发 2017-10-12 大家好,我是杰杰.       今晚,设计电路搞了一晚上,终于从模电渣渣的我,把MOS管理解了那么一丢丢,如有写的不好的地方,请指出来.谢谢.      ...

  3. spring中少用的注解@primary解析

    这次看下spring中少见的注解@primary注解,例子 @Component public class MetalSinger implements Singer{ @Override publi ...

  4. TensorFlow(十):卷积神经网络实现手写数字识别以及可视化

    上代码: import tensorflow as tf from tensorflow.examples.tutorials.mnist import input_data mnist = inpu ...

  5. 配置interfaces

    demo1 # This file describes the network interfaces available on your system # and how to activate th ...

  6. 前端性能测试工具Chrome performance

    页面加载速度慢,到底是多少秒,瓶颈在哪里? 前端性能工具Chrome performance 结合F12,基本可以搞定. 一.Chrome performance 1.shift+ctrl+N进入谷歌 ...

  7. [bzoj 5143][Ynoi 2018]五彩斑斓的世界

    传送门 Descroption 给了你一个长为n的序列a,有m次操作 1.把区间[l,r]中大于x的数减去x 2.查询区间[l,r]中x的出现次数 Solution 分块 对于每个块,我们都分别维护: ...

  8. Java学习日记基础(五)——类、对象之this、静态变量(类变量)、静态方法(类方法)、四大特征

    this 赵本山问奥尼尔:“我的爸爸的爸爸是谁?” 奥尼尔:“不知道” 赵本山:“你傻啊,是我爷爷” 奥尼尔回去问科比:“我的爸爸的爸爸是谁?” 科比:“不知道” 奥尼尔:”你傻啊,是赵本山的爷爷“ ...

  9. Apache Flink - Window

    Window: 在Streaming中,数据是无限且连续的,我们不可能等所有数据都到才进行处理,我们可以来一个就处理一下,但是有时我们需要做一些聚合类的处理,例如:在过去的1分钟内有多少用户点击了我们 ...

  10. bind--dns-docker---[nslookup/dig]

    [dig]  https://www.cnblogs.com/apexchu/p/6790241.html [dns resolution and revserse ]https://www.cnbl ...