
在开发基于 shiny for python 的交互式应用时,我们经常需要处理一些耗时的操作,例如通过串口发送一系列指令来控制外部设备。如果这些操作直接放在 @reactive.effect 或 @reactive.event 装饰器修饰的函数内部,并且包含了阻塞式的循环或长时间的延迟(如 time.sleep() 或忙等待 while 循环),就会导致整个 shiny 应用的用户界面(ui)失去响应。
考虑一个控制流体泵的场景:用户点击“启动”按钮(p1),应用开始按照预设的流量曲线循环发送串口指令。如果用户希望在传输过程中随时点击“停止”按钮(p2)来中断传输,那么一个阻塞式的启动逻辑将无法满足需求。原始实现中,p1 按钮对应的 _ 函数内部包含一个 while 循环,每次发送指令后都会等待两秒。这意味着在循环完成之前,p2 按钮的点击事件将无法被 Shiny 应用的主事件循环及时捕获和处理,导致停止指令被排队,直到当前传输循环结束后才能执行。
原始的阻塞式代码示例(存在响应性问题):
import time
import serial
from shiny import reactive
# 假设 ser 已经初始化为串口对象
ser = serial.Serial("COM6", 115200)
@reactive.Effect
@reactive.event(input.p1)
def _():
y = yg.get() # 从 reactive value yg 获取电压数组
for e in y: # 遍历数组
msg = "1:1:"+str(e)+":100" # 格式化驱动电压消息
ser.write(bytes(msg,'utf-8')) # 发送消息
t0 = time.time() # 记录时间戳
while(((time.time()-t0)<=2)): # 忙等待,直到2秒后
pass
ser.write(bytes("0:1",'utf-8')) # 传输结束后停止泵
@reactive.Effect
@reactive.event(input.p2)
def _():
#print("1:0")
ser.write(bytes("0:1",'utf-8')) # 停止泵问题分析: 上述 input.p1 对应的 _ 函数内部的 for 循环和 while 忙等待是导致问题的根源。在 Shiny 应用中,所有 reactive.Effect 和 reactive.event 装饰器修饰的函数都在同一个主线程中执行。当一个函数长时间运行时,它会独占主线程,阻止其他事件(如 input.p2 的点击)被处理,从而导致 UI 卡顿和失去响应。
为了解决主线程阻塞问题,我们可以将耗时操作从主线程中剥离,放到一个独立的后台线程中执行。Python 的 threading 模块提供了实现这一目标的工具,特别是 threading.Thread 用于创建新线程,以及 threading.Event 用于线程间的信号通信。
核心思路:
改进后的代码实现:
import serial
import time
import numpy as np
import threading as th
from shiny import App, ui, reactive
# 假设 ser 已经初始化
ser = serial.Serial("COM6", 115200)
# 定义一个全局的 Event 对象,用于线程间通信
sflag = th.Event()
# 辅助函数:发送串口消息
def transmit(e):
"""
根据给定的电压值 e 格式化消息并发送到串口。
"""
msg = "1:1:"+str(e)+":100"
# print(msg) # 调试用
ser.write(bytes(msg,'utf-8'))
# 后台线程执行的函数:定时发送数据
def rtimer(y, sflag):
"""
在独立线程中执行的函数,循环遍历数组 y 并发送数据。
每隔2秒发送一次,直到数组遍历完毕或 sflag 被设置。
"""
i = 0
while i < np.size(y) and not sflag.is_set():
transmit(y[i])
i += 1
time.sleep(2) # 使用 time.sleep() 在子线程中安全等待
# 循环结束后,如果不是因为 sflag 停止,则发送停止指令
# 但由于 p2 也会发送停止指令,此处可以根据实际需求调整
if not sflag.is_set(): # 如果是正常完成,而不是被中断
ser.write(bytes("0:1",'utf-8')) # 停止泵
# p1 按钮的响应函数:启动传输线程
@reactive.Effect()
@reactive.event(input.p1)
def start_pump_transmission():
"""
处理 p1 按钮点击事件,启动数据传输线程。
"""
y = yg.get() # 从 reactive value yg 获取数据
sflag.clear() # 启动前清除停止信号,确保线程可以运行
# 创建并启动新线程
timer_thread = th.Thread(target=rtimer, args=[y, sflag])
timer_thread.start()
# p2 按钮的响应函数:停止传输
@reactive.Effect()
@reactive.event(input.p2)
def stop_pump_transmission():
"""
处理 p2 按钮点击事件,设置停止信号并立即发送停止指令。
"""
sflag.set() # 设置停止信号,通知后台线程停止
ser.write(bytes("1:0",'utf-8')) # 立即发送停止泵的指令代码解释:
优点:
注意事项:
在 Shiny for Python 应用中,处理耗时或阻塞式操作的关键在于将其从主事件循环中分离。通过利用 Python 的 threading 模块,我们可以将这些任务放到独立的后台线程中执行,并使用 threading.Event 等机制进行线程间的有效通信,从而实现非阻塞的 UI 体验和对任务的精确控制。这种方法不仅解决了 UI 响应性问题,也使得应用能够更好地处理复杂的实时交互场景,如本例中对流体泵的即时启停控制。
以上就是如何在 Shiny 应用中处理长时间运行任务并保持 UI 响应性的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号