异步读取两个串行端口

时间:2016-05-29 00:39:43

标签: python pyserial uart

我想在Linux上的python中同时读取两个(或更多)串行端口(/ dev / ttyUSB0等)。我想从每个端口读取完整的行(以数据为准)并按接收的顺序处理结果(没有竞争条件)。一个简单的例子可以将行写入单个合并文件。

我认为这样做的方法是基于pyserial,但我无法弄清楚如何做到这一点。 Pyserial使用asyncio并使用threads进行非阻塞读取。 Asyncio被标记为实验性的。我假设如果在asyncio.Protocol.data_received()中完成处理,则不存在任何竞争条件。对于线程,处理可能必须由互斥锁保护。

也许这也可以不在pyserial中完成。两个串行端口可以作为文件打开,然后使用select()从数据可用时读取。

3 个答案:

答案 0 :(得分:1)

正如@AlexHall在评论中所建议的,这是一个解决方案,每个串口使用一个线程,同步访问队列:

import serial
import Queue
import threading

queue = Queue.Queue(1000)

def serial_read(s):
    while True:
        line = s.readline()
        queue.put(line)

serial0 = serial.Serial('/dev/ttyUSB0')
serial1 = serial.Serial('/dev/ttyUSB1')

thread1 = threading.Thread(target=serial_read, args=(serial0,),).start()
thread2 = threading.Thread(target=serial_read, args=(serial1,),).start()

while True:
    line = queue.get(True, 1)
    print line

有可能更优雅地写这个,但它有效。

答案 1 :(得分:1)

考虑使用aioserial

这是一个例子:

import asyncio
import concurrent.futures
import queue

import aioserial


async def readline_and_put_to_queue(
        aioserial_instance: aioserial.AioSerial,
        q: queue.Queue):
    while True:
        q.put(await aioserial_instance.readline_async())


async def process_queue(q: queue.Queue):
    with concurrent.futures.ThreadPoolExecutor(max_workers=1) as executor:
        while True:
            line: bytes = await asyncio.get_running_loop().run_in_executor(
                    executor, q.get)
            print(line.decode(errors='ignore'), end='', flush=True)
            q.task_done()


q: queue.Queue = queue.Queue()
aioserial_ttyUSB0: aioserial.AioSerial = \
        aioserial.AioSerial(port='/dev/ttyUSB0')
aioserial_ttyUSB1: aioserial.AioSerial = \
        aioserial.AioSerial(port='/dev/ttyUSB1', baudrate=115200)

asyncio.run(asyncio.wait([
    readline_and_put_to_queue(aioserial_ttyUSB0, q),
    readline_and_put_to_queue(aioserial_ttyUSB1, q),
    process_queue(q),
]))

答案 2 :(得分:0)

您可以尝试按顺序获取值并将其记忆在变量中:

a = data1.read ()
b = data2.read ()

按顺序处理后:

If len (a) != 0 or len (b ) != 0:
             Process a
             Process b

如果一个或两个值都有数据,请使用此方法处理