首页 > 编程语言 >QT设置回调函数给python调用——内置模块法

QT设置回调函数给python调用——内置模块法

时间:2024-08-16 15:23:56浏览次数:19  
标签:初始化 解释器 内置 QT callback python Python

1.QT 相关函数定义 和 QT设置回调函数给python调用——参数法中的定义相同如下:

// 实际的回调函数
void printValue(int value) {
    qDebug() << "printValue value: " << value;
}

int getValue(int value) {
    qDebug() << "getValue value: " << value;
    return value;
}

// 包装回调函数,使其成为Python可调用对象
PyObject* cprintValue(PyObject* self, PyObject* args) {
    int value;
    if (!PyArg_ParseTuple(args, "i", &value)) {
        return NULL;
    }
    printValue(value);
    return Py_None;
}

PyObject* cgetValue(PyObject* self, PyObject* args) {
    int value;
    if (!PyArg_ParseTuple(args, "i", &value)) {
        return NULL;
    }
    return Py_BuildValue("i", getValue(value));
}

// 模块方法定义
// 结构体第三个参数
// METH_VARARGS:方法接受可变数量的参数。
// METH_KEYWORDS:方法接受关键字参数。
// METH_NOARGS:方法不接受参数。
// METH_O:方法接受一个对象参数。
static PyMethodDef CallbackMethods[] = {
    {"cprintValue", cprintValue, METH_VARARGS, "printValue callback function"},
    {"cgetValue", cgetValue, METH_VARARGS, "getValue callback function"},
    {NULL, NULL, 0, NULL} //这一行为结束的标识
};

// 模块定义
static struct PyModuleDef CallbackModule = {
    PyModuleDef_HEAD_INIT,
    "callback",   // 模块名称
    NULL,         // 模块文档描述
    -1,           // 模块状态
    CallbackMethods, //模块中的函数和方法
    nullptr, //指向一个 PyModuleDef_Slot 结构体数组的指针,用于定义模块的特殊属性
    nullptr, //指向一个函数指针,用于遍历模块对象的函数
    nullptr, //指向一个函数指针,用于清除模块对象的函数
    nullptr //指向一个函数指针,用于释放模块对象的函数
};

// 模块初始化函数
PyMODINIT_FUNC PyInit_callback(void) {
    PyObject *m = PyModule_Create(&CallbackModule);
    if(m == NULL){
        qDebug("PyInit_callback failed!");
        return NULL;
    }
    return m;
}

2. python 相关函数定义

# python_script_2.py

import callback

def printValue():
    callback.cprintValue(45)

def getValue():
    callback.cgetValue(256)

3. QT 测试

int main(int argc, char *argv[])
{
    QCoreApplication app(argc, argv);
    // 设置Python解释器的路径
    wchar_t *program = Py_DecodeLocale("python38_64", nullptr);
    Py_SetProgramName(program);
    Py_SetPath(L"D:\\Python\\Python38_64;D:\\Python\\Python38_64\\DLLs;D:\\Python\\Python38_64\\lib");

    // 添加自定义模块
    PyImport_AppendInittab("callback", PyInit_callback);

    // 初始化Python解释器
    Py_Initialize();

    PyRun_SimpleString("import sys");
    PyRun_SimpleString("sys.path.append('./')");

    PyObject *demo = PyImport_ImportModule("python_script_2");
    if(demo == NULL){
        qInfo() << "demo is nullptr!";
        PyErr_Print();
        return -1;
    }

    PyObject *pyPrintValue = PyObject_GetAttrString(demo,"printValue");
    if(pyPrintValue == NULL){
        qInfo() << "pyPrintValue is nullptr!";
        return -1;
    }
    PyObject* pReturnValue;
    PyObject* args = PyTuple_New(0);
    pReturnValue = PyObject_CallObject(pyPrintValue, args);

    PyObject *pyGetValue = PyObject_GetAttrString(demo,"getValue");
    if(pyGetValue == NULL){
        qInfo() << "pgetValue is nullptr!";
        return -1;
    }
    PyObject* pReturnValue2;
    PyObject* args2 = PyTuple_New(0);
    pReturnValue2 = PyObject_CallObject(pyGetValue, args2);

    Py_DECREF(demo);
    // 结束Python解释器
    Py_Finalize();

    return app.exec();
}

 说明:

PyImport_AppendInittab 函数通常在 Python 解释器初始化之前调用。它的作用是注册一个 C 函数,以便在 Python 解释器启动时初始化一个新的内置模块。因此,你需要在 Python 解释器初始化之前调用 PyImport_AppendInittab 函数,以确保你的自定义模块能够正确初始化

相当于使用C++/QT写一个内置的python模块在初始化之前内置到解释器,后面的模块都可以使用。

标签:初始化,解释器,内置,QT,callback,python,Python
From: https://blog.csdn.net/weixin_39344648/article/details/141227878

相关文章

  • Python借助Selenium,保留原样式拷贝网站资源
    importurllib3fromseleniumimportwebdriverfrombs4importBeautifulSoupimportosimportrequestsfromurllib.parseimporturljoin,urlparseimportbase64importrefromrequests.adaptersimportHTTPAdapterfromrequests.packages.urllib3.util.retryim......
  • Python魔法应用:自动化日常任务:让Python成为你的私人助理
    自动化日常任务:让Python成为你的私人助理引言自动化文本处理:批量重命名文件、整理数据批量重命名文件整理数据自动化邮件发送:使用smtplib发送邮件自动化网页交互:使用requests库获取网页数据结语引言在日常工作和学习中,我们常常会遇到大量重复且繁琐的任务,这些任务......
  • python 之 pytest框架
    一、pytest单元测试框架1、什么是单元测试,针对软件的最小单位(函数,方法,接口)进行正确性的检查测试。 2、单元测试框架(按语言分)java:junit和testingpython:unittest和pytest 3、单元测试主要是做什么呢?(1)、测试发现:从多个文件里面去找到测试用例(2)、测试执行:按......
  • python蟒蛇绘制
    turtle海龟绘图setup()设置窗体大小及位置setup()不是必须的importturtleturtle.setup(650,350,200,200)turtle.penup()turtle.fd(-250)turtle.pendown()turtle.pensize(25)turtle.pencolor("purple")turtle.seth(-40)foriinrange(4):turtle.circle(40,80)......
  • Python温度的转换
    神清气爽哈哈哈程序代码:#温度的转换TempStr=input("请输入带有符号的温度值:")ifTempStr[-1]in['F','f']:#判断字符转最后一个字符是否为F或fc=(eval(TempStr[0:-1])-32)/1.8#将摄氏度转化为华氏度print("转换后的温度是{:.2f}C".format(c))#eva......
  • 【Python SHA256 摘要算法】
    SHA256是一种广泛使用的密码散列函数,用于生成数据的唯一指纹,即散列值。它属于SHA-2家族,该家族还包括SHA-384和SHA-512算法。SHA256算法在许多领域都有应用,例如:数据完整性验证:用于验证数据在传输或存储过程中是否被篡改。例如,在下载软件时,通常会提供软件的SHA256哈......
  • 使用Python创建省份城市地图选择器
    在这篇博客中,我们将探讨如何使用Python创建一个简单而实用的省份城市地图选择器。这个项目不仅能帮助我们学习Python的基础知识,还能让我们了解如何处理JSON数据和集成网页浏览器到桌面应用程序中。C:\pythoncode\new\geographicgooglemap.py全部代码importwximportwx.......
  • 【Python-办公自动化】1秒提取PPT文本内容形成目录保存至WORD
    欢迎来到"花花ShowPython",一名热爱编程和分享知识的技术博主。在这里,我将与您一同探索Python的奥秘,分享编程技巧、项目实践和学习心得。无论您是编程新手还是资深开发者,都能在这里找到有价值的信息和灵感。自我介绍:我热衷于将复杂的技术概念以简单易懂的方式呈现给大家,......
  • 【Python-办公自动化】几分钟生成上万份合同(用此思路可以用于写作固定格式的文章)
    欢迎来到"花花ShowPython",一名热爱编程和分享知识的技术博主。在这里,我将与您一同探索Python的奥秘,分享编程技巧、项目实践和学习心得。无论您是编程新手还是资深开发者,都能在这里找到有价值的信息和灵感。自我介绍:我热衷于将复杂的技术概念以简单易懂的方式呈现给大家,......
  • SSM-国外鞋服代购平台-97782(免费领源码+开发文档)可做计算机毕业设计JAVA、PHP、爬虫、
    SSM国外鞋服代购平台摘 要随着科学技术的飞速发展,社会的方方面面、各行各业都在努力与现代的先进技术接轨,通过科技手段来提高自身的优势,鞋服代购平台当然也不例外。代购平台是以实际运用为开发背景,运用软件工程原理和开发方法,采用Java技术构建的一个管理系统。整个开发过......