当前位置: 首页 > news >正文

【Qt】bug排查笔记——QMetaObject::invokeMethod: No such method

问题如题目所示:QMetaObject::invokeMethod: No such method xxxx,在网上好一顿查,又将查到的资料喂给了 Ai,才最终将问题解决,特此记录下。

一、问题背景

在做公司项目时,使用了插件的方式开发。主程序加载了一个叫CtmAboutAppPlugin的插件。该插件有个界面类CtmAboutUi,它有个槽函数:

public slot:void subAppInfo(const QVariant& msg);

主程序通过一个叫 CbbEventBus的事件总线机制,把消息发送给CtmAboutAppPlugin 这个插件。

此处我给出CbbEventBus的订阅方法的源码(问题出在了订阅方法这里):

bool CbbEventBus::subscribe(const QString &topic, QObject *receiver, const char *method, Qt::ConnectionType type)
{if (!receiver || !method) return false;auto connection = QObject::connect(this,&CbbEventBus::signalUpdateMessage,receiver,[receiver, method, topic](const QString &receivedTopic, const QVariant &msg) {if (receivedTopic == topic) {// 问题出现在下面这里// QMetaObject::invokeMethod(receiver, method, Q_ARG(const QVariant&, msg)); // 最开始我是这么写的QMetaObject::invokeMethod(receiver, method, Q_ARG(QVariant, msg)); 	// 后来改成了左边这样}},type);if (connection) {m_QIsSubscribeMap[topic][receiver].append(connection);return true;}return false;
}

上面两种写法,在程序运行时,控制台分别输出以下信息:

Q_ARG(QVariant, msg)对应控制台输出信息:QMetaObject::invokeMethod: No such method CtmAboutUi::1subAppInfo(QVariant)(const QVariant&)

Q_ARG(QVariant, msg)对应控制台输出信息:No such method CtmAboutUi::1subAppInfo(QVariant)(QVariant)

二、🔍 问题原因(一句话总结)

根本原因不是方法不存在,而是 invokeMethod 找方法时用的名字“对不上号”——我传了个“带参数的签名”,它却以为这是“方法名”,导致匹配失败。


三、🕵️‍♂️ 排查过程回顾

3.1 第一反应:是不是 MOC 没生效?

○ 检查了 CtmAboutUi 类,Q_OBJECT 有,public slots: 有,语法没问题。
○ 打开 Qt 生成的 moc_CtmAboutUi.cpp一看,MOC 确实生成了,subAppInfo(QVariant) 也注册进去了,说明元对象系统这块是没有问题的。
在这里插入图片描述

3.2 第二反应:是不是 Q_ARG 写错了?

○ 一开始用了 Q_ARG(const QVariant&, msg),这是个经典坑。
○ Qt 的 Q_ARG 第一个参数是类型名,不能带 const&(否则元对象系统会直接按照 const QVariant&去匹配字符串,实际上元对象系统中注册的是QVariant 类型,并没有const QVariant&),应该写成 Q_ARG(QVariant, msg)
○ 改了之后,错误还在,但变成了 (QVariant)(QVariant),说明问题没完。

3.3 第三反应:名字到底传了啥?

eventBus.subscribe(topic, m_pAboutUi, SLOT(subAppInfo(QVariant)));

○ 这里 SLOT(...) 宏展开后是 "subAppInfo(QVariant)",是个带参数列表的字符串。
○ 而 invokeMethod 拿到这个字符串后,会把它当“方法名”去查,再配上 Q_ARG(QVariant, msg),就变成了"subAppInfo(QVariant)" + “(QVariant)”,即下面这样:

subAppInfo(QVariant)(QVariant)

这当然找不到,因为实际注册的是 subAppInfo(QVariant)

四、✅ 最终解决办法

把调用方式从:

eventBus.subscribe(this->topic(), m_pAboutUi, SLOT(subAppInfo(QVariant)));

改成:

eventBus.subscribe(this->topic(), m_pAboutUi, "subAppInfo");

只传方法名,不带参数列表。这样 invokeMethod 就会用方法名 “subAppInfo” 去找,再根据 Q_ARG(QVariant, msg) 匹配参数类型,完美匹配成功。


4.1 其它疑问:

我也尝试了如下方法:

虽然 "subAppInfo" 能解决问题,但更推荐用 函数指针 的方式,既安全又现代:


eventBus.subscribe(this->topic(), m_pAboutUi, &CtmAboutUi::subAppInfo);

这需要 CbbEventBus 支持模板,但好处是:
● 编译时检查,名字写错直接报错
● 不用拼字符串,不怕类型不匹配
● IDE 能跳转,维护方便

但是有错误,没有成功…


五、📝 总结

QMetaObject::invokeMethod: No such method 不一定是方法不存在,很可能是 名字传错了。
SLOT() 宏返回的是带参数的字符串,不适合直接传给 invokeMethod 当方法名用。
Q_ARG 只写类型名,别带 const&
个人认为最稳妥的方式是用函数指针 &Class::method,但是该方法没有成功,由于工作时间问题,目前还没继续深究…

http://www.dtcms.com/a/363077.html

相关文章:

  • 面试_Mysql
  • AdaBoost(Adaptive Boosting,自适应提升算法)总结梳理
  • 04 创建Centos 7操作系统
  • 基于ZooKeeper实现分布式锁(Spring Boot接入)及与Kafka实现的对比分析
  • 【Vue2 ✨】 Vue2 入门之旅(六):指令与过滤器
  • React 中 key 的作用
  • Rust SQLx 开发指南:利用 Tokio 进行性能优化
  • Spring Security资源服务器在高并发场景下的认证性能优化实践指南
  • FPGA AD7606串行驱动与并行驱动
  • AI如何理解PDF中的表格和图片?
  • 【HarmonyOS 6】仿AI唤起屏幕边缘流光特效
  • 使用Java获取本地PDF文件并解析数据
  • Echarts自定义横向柱状图中单条bar的样式
  • 从模态融合到高效检索:微算法科技 (NASDAQ:MLGO)CSS场景下的图卷积哈希方法全解析
  • 九月科技瞭望:中国科技发展规划动态洞察
  • DevExpress WPF中文教程:如何将WPF数据网格绑定到本地数据库?
  • Python 2025:量子计算、区块链与边缘计算的新前沿
  • [Linux]学习笔记系列 -- mm/swap.c 交换机制(Swap Mechanism) 物理内存的虚拟扩展
  • Linux92 shell:倒计时,用户分类
  • 【JavaEE】多线程案例
  • 删除⽂件之git
  • 前端20个高效开发的JS工具函数
  • 《水浒智慧》第二部“英雄是怎么炼成的”(下篇)读书笔记
  • 宋红康 JVM 笔记 Day11|直接内存
  • 怎么用redis lua脚本实现各分布式锁?Redisson各分布式锁怎么实现的?
  • Higress云原生API网关详解 与 Linux版本安装指南
  • lua脚本在redis中如何单步调试?
  • docker 安装 redis 并设置 volumes 并修改 修改密码(二)
  • MATLAB矩阵及其运算(四)矩阵的运算及操作
  • 互联网大厂求职面试记:谢飞机的搞笑答辩