0

0

保持Python脚本关闭后对象状态的方法

聖光之護

聖光之護

发布时间:2025-08-23 16:00:18

|

721人浏览过

|

来源于php中文网

原创

保持python脚本关闭后对象状态的方法

本文旨在解决在LabVIEW调用Python脚本控制电子板时,如何保持电子板对象状态,避免频繁开关串口导致连接问题。文章将探讨通过后台运行脚本或进程,以及在关闭串口前清理缓冲区和增加延时等方法,确保串口连接的稳定性和可靠性。

在LabVIEW等环境中调用Python脚本控制硬件设备,例如电子板时,经常会遇到串口通信的问题。一个常见的场景是,每次调用一个Python脚本,都会初始化电子板对象并打开串口,脚本执行完毕后关闭串口。这种频繁的开关操作可能会导致串口连接不稳定,甚至出现端口占用等问题。本文将介绍几种解决此问题的方法,以确保串口通信的稳定性和可靠性。

方法一:后台运行脚本或进程

一种解决方案是将初始化电子板对象的脚本作为一个独立的进程或服务在后台运行。这样,电子板对象只会被初始化一次,串口也只会被打开一次,避免了频繁的开关操作。

  1. 创建初始化脚本: 创建一个Python脚本,用于初始化电子板对象并保持运行状态。

    立即学习Python免费学习笔记(深入)”;

    # Set_Board.py
    import time
    import serial
    
    class ElectronicBoard:
        def __init__(self, com_port="COM5", verbose=True):
            self.com_port = com_port
            self.verbose = verbose
            try:
                self.ser = serial.Serial(self.com_port, 9600, timeout=1) # 串口初始化
                self.is_powered = True
                if self.verbose:
                    print("Connected!")
            except serial.SerialException as e:
                self.is_powered = False
                print(f"Connection failed: {e}")
                self.ser = None
    
        def doFunctionX(self):
            if self.ser:
                self.ser.write(b"X")  # 发送指令X
                print("Function X executed")
            else:
                print("Serial port not initialized")
    
        def doFunctionY(self):
            if self.ser:
                self.ser.write(b"Y")  # 发送指令Y
                print("Function Y executed")
            else:
                print("Serial port not initialized")
    
        def close(self):
            if self.ser:
                self.ser.close()
                print("Serial port closed")
            else:
                print("Serial port not initialized")
    
    board = ElectronicBoard(com_port="COM5", verbose=True)
    
    if board.is_powered:
        print("Board initialized and running in the background.")
        while True:
            time.sleep(1)  # 保持脚本运行
    else:
        print("Failed to initialize board.")
  2. 后台运行脚本: 可以使用各种方法在后台运行此脚本,例如使用nohup命令(Linux)或将其转换为Windows服务。

    • Linux (nohup):

      nohup python Set_Board.py &
    • Windows (创建服务): 可以使用pywin32库将Python脚本转换为Windows服务。

  3. 其他脚本调用: 其他脚本可以通过某种进程间通信(IPC)机制与后台运行的脚本进行通信,例如使用套接字、管道或消息队列。

    宣小二
    宣小二

    宣小二:媒体发稿平台,自媒体发稿平台,短视频矩阵发布平台,基于AI驱动的企业自助式投放平台。

    下载
    # Script1.py
    import socket
    
    def send_command(command):
        with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
            s.connect(('localhost', 65432))  # 连接到后台脚本的套接字
            s.sendall(command.encode())
            data = s.recv(1024)
        print(f"Received {data.decode()}")
    
    send_command("doFunctionX")
    # Set_Board.py (修改后的后台脚本)
    import socket
    import threading
    
    # ... (ElectronicBoard 类定义)
    
    board = ElectronicBoard(com_port="COM5", verbose=True)
    
    def handle_connection(conn, addr):
        with conn:
            print(f"Connected by {addr}")
            while True:
                data = conn.recv(1024)
                if not data:
                    break
                command = data.decode()
                print(f"Received command: {command}")
                if command == "doFunctionX":
                    board.doFunctionX()
                    conn.sendall(b"Function X executed")
                elif command == "doFunctionY":
                    board.doFunctionY()
                    conn.sendall(b"Function Y executed")
                else:
                    conn.sendall(b"Unknown command")
    
    with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
        s.bind(('localhost', 65432))
        s.listen()
        print("Board initialized and listening for commands.")
        while True:
            conn, addr = s.accept()
            thread = threading.Thread(target=handle_connection, args=(conn, addr))
            thread.start()
    

    注意事项: 需要选择合适的IPC机制,并确保其稳定性和安全性。

方法二:清理缓冲区和增加延时

如果无法使用后台进程,可以尝试在每次关闭串口之前,清理输入和输出缓冲区,并在关闭串口后增加一个延时。

# Script1.py
import serial
import time

try:
    board = serial.Serial("COM5", 9600, timeout=1)
    print("Connected!")
except serial.SerialException as e:
    print(f"Connection failed: {e}")
    board = None

if board:
    board.write(b"X")  # 发送指令X
    print("Function X executed")

    board.flushInput()  # 清理输入缓冲区
    board.flushOutput() # 清理输出缓冲区
    board.close()
    print("Serial port closed")
    time.sleep(2)  # 增加延时 (至少1-2秒)

解释:

  • board.flushInput(): 清空输入缓冲区,丢弃所有未读取的数据。
  • board.flushOutput():清空输出缓冲区,等待所有数据发送完毕。
  • time.sleep(2): 在关闭串口后暂停2秒。这个延时允许操作系统释放串口资源,避免端口被立即占用。

注意事项:

  • 延时时间需要根据实际情况调整。
  • 确保在所有脚本中都使用相同的清理缓冲区和延时策略。

方法三:使用单例模式

可以使用单例模式来确保只有一个电子板对象存在。这样,所有的脚本都使用同一个对象,避免重复初始化和关闭串口。

# ElectronicBoardSingleton.py
import serial

class ElectronicBoard:
    _instance = None

    def __new__(cls, com_port="COM5", verbose=True):
        if cls._instance is None:
            cls._instance = super(ElectronicBoard, cls).__new__(cls)
            cls._instance.com_port = com_port
            cls._instance.verbose = verbose
            try:
                cls._instance.ser = serial.Serial(cls._instance.com_port, 9600, timeout=1)
                cls._instance.is_powered = True
                if cls._instance.verbose:
                    print("Connected!")
            except serial.SerialException as e:
                cls._instance.is_powered = False
                print(f"Connection failed: {e}")
                cls._instance.ser = None
        return cls._instance

    def doFunctionX(self):
        if self.ser:
            self.ser.write(b"X")  # 发送指令X
            print("Function X executed")
        else:
            print("Serial port not initialized")

    def doFunctionY(self):
        if self.ser:
            self.ser.write(b"Y")  # 发送指令Y
            print("Function Y executed")
        else:
            print("Serial port not initialized")

    def close(self):
        if self.ser:
            self.ser.close()
            print("Serial port closed")
            self.ser = None  # 确保下次调用时重新初始化
        else:
            print("Serial port not initialized")
# Script1.py
from ElectronicBoardSingleton import ElectronicBoard

board = ElectronicBoard(com_port="COM5")
board.doFunctionX()
board.close()
# Script2.py
from ElectronicBoardSingleton import ElectronicBoard

board = ElectronicBoard(com_port="COM5")
board.doFunctionY()
board.close()

注意事项:

  • 单例模式确保只有一个对象,但仍然需要在每次脚本结束后关闭串口,并清理缓冲区和增加延时,以避免端口占用问题。
  • 在多线程或多进程环境下使用单例模式需要注意线程安全问题。

总结

本文介绍了三种解决在LabVIEW调用Python脚本控制电子板时,保持电子板对象状态的方法:后台运行脚本或进程、清理缓冲区和增加延时、以及使用单例模式。选择哪种方法取决于具体的应用场景和需求。如果需要频繁地与电子板进行通信,并且对实时性要求较高,建议使用后台运行脚本或进程。如果只是偶尔需要与电子板进行通信,并且可以容忍一定的延时,可以使用清理缓冲区和增加延时的方法。使用单例模式可以确保只有一个电子板对象存在,但仍然需要注意串口资源的释放。 无论选择哪种方法,都需要进行充分的测试,以确保串口通信的稳定性和可靠性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

546

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

212

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

20

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

19

2026.01.21

windows查看端口占用情况
windows查看端口占用情况

Windows端口可以认为是计算机与外界通讯交流的出入口。逻辑意义上的端口一般是指TCP/IP协议中的端口,端口号的范围从0到65535,比如用于浏览网页服务的80端口,用于FTP服务的21端口等等。怎么查看windows端口占用情况呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

859

2023.07.26

查看端口占用情况windows
查看端口占用情况windows

端口占用是指与端口关联的软件占用端口而使得其他应用程序无法使用这些端口,端口占用问题是计算机系统编程领域的一个常见问题,端口占用的根本原因可能是操作系统的一些错误,服务器也可能会出现端口占用问题。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1133

2023.07.27

windows照片无法显示
windows照片无法显示

当我们尝试打开一张图片时,可能会出现一个错误提示,提示说"Windows照片查看器无法显示此图片,因为计算机上的可用内存不足",本专题为大家提供windows照片无法显示相关的文章,帮助大家解决该问题。

806

2023.08.01

windows查看端口被占用的情况
windows查看端口被占用的情况

windows查看端口被占用的情况的方法:1、使用Windows自带的资源监视器;2、使用命令提示符查看端口信息;3、使用任务管理器查看占用端口的进程。本专题为大家提供windows查看端口被占用的情况的相关的文章、下载、课程内容,供大家免费下载体验。

455

2023.08.02

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

30

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.3万人学习

Git 教程
Git 教程

共21课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号