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

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

一、问题背景

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

cpp 复制代码
public slot:
    void subAppInfo(const QVariant& msg);

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

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

cpp 复制代码
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 第三反应:名字到底传了啥?

cpp 复制代码
eventBus.subscribe(topic, m_pAboutUi, SLOT(subAppInfo(QVariant)));

○ 这里 SLOT(...) 宏展开后是 "subAppInfo(QVariant)",是个带参数列表的字符串。

○ 而 invokeMethod 拿到这个字符串后,会把它当"方法名"去查,再配上 Q_ARG(QVariant, msg),就变成了"subAppInfo(QVariant)" + "(QVariant)",即下面这样:

cpp 复制代码
subAppInfo(QVariant)(QVariant)

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

四、✅ 最终解决办法

把调用方式从:

cpp 复制代码
eventBus.subscribe(this->topic(), m_pAboutUi, SLOT(subAppInfo(QVariant)));

改成:

cpp 复制代码
eventBus.subscribe(this->topic(), m_pAboutUi, "subAppInfo");

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


4.1 其它疑问:

我也尝试了如下方法:

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

cpp 复制代码
eventBus.subscribe(this->topic(), m_pAboutUi, &CtmAboutUi::subAppInfo);

这需要 CbbEventBus 支持模板,但好处是:

● 编译时检查,名字写错直接报错

● 不用拼字符串,不怕类型不匹配

● IDE 能跳转,维护方便

但是有错误,没有成功...


五、📝 总结

QMetaObject::invokeMethod: No such method 不一定是方法不存在,很可能是 名字传错了。

SLOT() 宏返回的是带参数的字符串,不适合直接传给 invokeMethod 当方法名用。

Q_ARG 只写类型名,别带 const&

个人认为最稳妥的方式是用函数指针 &Class::method,但是该方法没有成功,由于工作时间问题,目前还没继续深究...

相关推荐
青草地溪水旁1 小时前
设计模式(C++)详解—享元模式(1)
c++·设计模式·享元模式
源代码•宸1 小时前
GAMES101:现代计算机图形学入门(Chapter2 向量与线性代数)迅猛式学线性代数学习笔记
经验分享·笔记·学习·线性代数·计算机图形学
雪域迷影1 小时前
使用C++编写的一款射击五彩敌人的游戏
开发语言·c++·游戏
郝学胜-神的一滴2 小时前
享元模式(Flyweight Pattern)
开发语言·前端·c++·设计模式·软件工程·享元模式
charlie1145141912 小时前
精读《C++20设计模式》——创造型设计模式:构建器系列
c++·设计模式·c++20·构造器模式
Larry_Yanan2 小时前
QML学习笔记(四)QML新手入门其二:通过MouseArea让Rectangle实现鼠标三态
笔记·qt·ui
小王努力学编程2 小时前
brpc远程过程调用
linux·服务器·c++·分布式·rpc·protobuf·brpc
青草地溪水旁3 小时前
设计模式(C++)详解—享元模式(2)
c++·设计模式·享元模式
郝学胜-神的一滴3 小时前
QT与Spring Boot通信:实现HTTP请求的完整指南
开发语言·c++·spring boot·后端·qt·程序人生·http
Ethan learn English3 小时前
English Around the House and Farm
笔记·英语·可理解性输入