QT多线程调用python

由于Python解释器有全局解释所GIL的原因,导致在同一时刻只能有一个线程拥有解释器,所以在C++多线程调用python脚本时,需要控制GIL,线程获取GIL。
所以一个程序里,无论有多少条线程调用python,python只能初始化一次。
每次线程要调用python时,都要拥有GIL。

//python只需要初始化一次
void PythonHandler::PythonInit()
{
    if( !Py_IsInitialized() )
    {
        //1.初始化Python解释器,这是调用操作的第一步
        Py_Initialize();
        if( !Py_IsInitialized() ){
            qDebug("初始化Python解释器失败");
         }else{
            // 初始化线程支持
           PyEval_InitThreads();
           // 启动子线程前执行,为了释放PyEval_InitThreads获得的全局锁,否则子线程可能无法获取到全局锁。
           PyEval_ReleaseThread(PyThreadState_Get());
           qDebug("初始化Python解释器成功");
        }
    }
}

//将全局解释器锁和线程的相关操作用类封装
#ifndef PYTHREADSTATELOCK_H
#define PYTHREADSTATELOCK_H
#include "Python.h"

class PyThreadStateLock
{
public:    
    PyThreadStateLock(void)
    {
        _save = nullptr;
        nStatus = 0;
        nStatus = PyGILState_Check() ;   //检测当前线程是否拥有GIL
        PyGILState_STATE gstate;
        if ( !nStatus )
        {
            gstate = PyGILState_Ensure();   //如果没有GIL,则申请获取GIL
            nStatus = 1;
        }
        _save = PyEval_SaveThread();
        PyEval_RestoreThread(_save);
    }
    ~PyThreadStateLock(void)
    {
        _save = PyEval_SaveThread();
        PyEval_RestoreThread(_save);
        if (nStatus)
        {
            PyGILState_Release(gstate);    //释放当前线程的GIL
        }
    }

private:
    PyGILState_STATE gstate;
    PyThreadState *_save;
    int nStatus;

};

#endif // PYTHREADSTATELOCK_H

然后我们就可以在子线程调用python

//加载模块
void PythonHandler::PythonInsmod(QString pyfname)
{
    class PyThreadStateLock PyThreadLock;//获取全局锁

    // Import
    if ( -1 == PyRun_SimpleString("import os") ) {
        qCritical()<<"Python缺少OS库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import math") ) {
        qCritical()<<"Python缺少Math库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import sys") ) {
        qCritical()<<"Python缺少Sys库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import time") ) {
        qCritical()<<"Python缺少Time库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("from PIL import Image") ) {
        qCritical()<<"Python缺少PIL库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import tensorflow as tf") ) {
        qCritical()<<"Python缺少Tensorflow库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import numpy as np") ) {
        qCritical()<<"Python缺少Numpy库";
        return ;
    }
    if ( -1 == PyRun_SimpleString("import cv2") ) {
        qCritical()<<"Python缺少OpenCV库";
        return ;
    }

    // 导入py模块
 	m_module = PyImport_ImportModule(pyfname.toLocal8Bit().data());

    if ( nullptr != m_module ) {
        qDebug()<<"加载Python文件成功";
    } else {
        qCritical()<<"加载Python文件失败"<<pyfname;
        return ;
    }

    // 载入方法
    m_detection = PyObject_GetAttrString(m_module, "detection");
    if ( nullptr != m_detection && PyCallable_Check( m_detection ) ) {
        qDebug()<<"加载Python方法成功";
    } else {
        qCritical()<<"加载Python方法失败";
        return ;
    }
}

后续,就可以在子线程尽情地调用python函数了
本文参考:C++ 多线程调用Python脚本

  • 4
    点赞
  • 39
    收藏
    觉得还不错? 一键收藏
  • 10
    评论
Qt提供了丰富的多线程编程支持,包括构造了一些基本的与平台无关的线程类、提交用户自定义事件的Thread-safe方式、多种线程间同步机制,如信号量、全局锁等。同时,Qt还提供了QTimer类,用于提供重复和单次定时器。下面是关于Qt多线程和计时器的介绍和演示: 1. Qt多线程的使用方法 Qt中的多线程编程主要通过QThread类实现。使用QThread类的步骤如下: (1)创建一个继承自QThread的子类,并重写其run()函数,run()函数中编写子线程的代码逻辑。 (2)在主线程中创建子线程对象,并调用其start()函数启动子线程。 下面是一个简单的例子,演示了如何使用QThread类创建一个子线程: ```python from PyQt5.QtCore import QThread, pyqtSignal class MyThread(QThread): # 定义一个信号,用于在子线程中发送消息给主线程 signal = pyqtSignal(str) def __init__(self, parent=None): super().__init__(parent) def run(self): # 子线程的代码逻辑 for i in range(5): self.signal.emit('当前进度:%d' % i) self.sleep(1) # 在主线程中创建子线程对象,并启动子线程 thread = MyThread() thread.signal.connect(lambda msg: print(msg)) thread.start() ``` 在上面的例子中,我们创建了一个继承自QThread的子类MyThread,并重写了其run()函数。在run()函数中,我们编写了子线程的代码逻辑,即每隔1秒钟向主线程发送一个消息。在主线程中,我们创建了子线程对象,并通过connect()函数将子线程的信号与一个lambda函数连接起来,当子线程发送信号时,lambda函数会被调用,输出消息。 2. QTimer的使用方法 QTimer类用于提供重复和单次定时器。使用QTimer类的步骤如下: (1)创建一个QTimer对象,并设置定时器的时间间隔。 (2)连接定时器的timeout()信号到一个槽函数中,槽函数中编写定时器触发时的代码逻辑。 (3)调用定时器的start()函数启动定时器。 下面是一个简单的例子,演示了如何使用QTimer类创建一个定时器: ```python from PyQt5.QtCore import QTimer # 创建一个QTimer对象,并设置定时器的时间间隔为1秒 timer = QTimer() timer.setInterval(1000) # 连接定时器的timeout()信号到一个槽函数中 def on_timeout(): print('定时器触发了') timer.timeout.connect(on_timeout) # 启动定时器 timer.start() ``` 在上面的例子中,我们创建了一个QTimer对象,并设置了定时器的时间间隔为1秒。然后,我们连接了定时器的timeout()信号到一个槽函数on_timeout()中,当定时器触发时,槽函数中的代码逻辑会被执行。最后,我们调用了定时器的start()函数启动定时器。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值