C++多线程中调用python api函数

今天看了近一天关于多线程的应用中,如何安全调用python方面的资料,开始的时候看的简直头大如斗,被python语言的全局锁(Global Interpreter Lock)、线程状态(Thread State )等都有点绕晕了,后来经过各方面文章和帮助文档的相互参考,发现对于2.4/2.5版本,提供了PyGILState_Ensure, PyGILState_Release,哎,这下可方便大发了。

一、首先定义一个封装类,主要是保证PyGILState_Ensure, PyGILState_Release配对使用,而且这个类是可以嵌套使用的。

#include <python.h>

class PyThreadStateLock
{
public:
    PyThreadStateLock(void)
    {
        state = PyGILState_Ensure( );
    }

    ~PyThreadStateLock(void)
    {
         PyGILState_Release( state );
    }
private:
    PyGILState_STATE state;
};


二、在主线程中,这样处理

    // 初始化
    Py_Initialize();
    // 初始化线程支持
    PyEval_InitThreads();
    // 启动子线程前执行,为了释放PyEval_InitThreads获得的全局锁,否则子线程可能无法获取到全局锁。
    PyEval_ReleaseThread(PyThreadState_Get());
   
    // 其他的处理,如启动子线程等
    ......
       
    // 保证子线程调用都结束后
    PyGILState_Ensure();
    Py_Finalize();
    // 之后不能再调用任何python的API

三、在主线程,或者子线程中,调用python本身函数的都采用如下处理

    {
        class PyThreadStateLock PyThreadLock;
        // 调用python的API函数处理
        ......
    }

呵呵,看这样是否非常简单了。


另外还有两个和全局锁有关的宏,Py_BEGIN_ALLOW_THREADS 和 Py_END_ALLOW_THREADS。这两个宏是为了在较长时间的C函数调用前,临时释放全局锁,完成后重新获取全局锁,以避免阻塞其他python的线程继续运行。这两个宏可以这样调用

    {
        class PyThreadStateLock PyThreadLock;
        // 调用python的API函数处理
        ......

        Py_BEGIN_ALLOW_THREADS
        // 调用需要长时间的C函数
        ......
        Py_END_ALLOW_THREADS

        // 调用python的API函数处理
        ......
    }
  • 0
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
Python,可以使用thread模块或者threading模块来实现多线程编程。在多线程编程,每个线程都有自己的代码执行流程和堆栈空间,但是它们共享进程的资源,如全局变量、文件、socket等。 如果要在多线程调用Python子解释器,可以使用Py_NewInterpreter()函数来创建一个新的解释器对象,并在这个解释器对象执行Python代码。 以下是一个示例代码: ``` import threading import ctypes import sys # 定义一个线程函数 def thread_func(): # 创建一个新的解释器对象 ctypes.pythonapi.Py_NewInterpreter() # 在新的解释器对象执行Python代码 exec("print('Hello from new interpreter')") # 释放解释器对象 ctypes.pythonapi.Py_EndInterpreter() # 在主线程创建一个新的子线程 t = threading.Thread(target=thread_func) t.start() # 在主线程执行Python代码 print('Hello from main interpreter') ``` 在上面的代码,我们首先导入了threading模块和ctypes模块。然后定义了一个线程函数thread_func(),在这个函数创建了一个新的解释器对象,并在这个解释器对象执行了一段Python代码。最后,在主线程创建了一个新的子线程,执行了一段Python代码,并输出了一些信息。 注意,在使用Py_NewInterpreter()函数创建一个新的解释器对象后,我们需要在新的解释器对象执行一些Python代码,否则这个解释器对象会很快被释放。在上面的代码,我们使用了exec()函数来执行一段简单的Python代码,你可以根据自己的需要来编写更复杂的代码。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值