qml调用c++类内函数的三种方法
一.方法一:使用 Q_INVOKABLE 宏声明成员函数
1.第一步:依然需要新建一个类NetworkHandler:
#include <QObject>
class NetworkHandler : public QObject
{
Q_OBJECT
public:
explicit NetworkHandler(QObject *parent = nullptr);
Q_INVOKABLE void onConnetClicked();
Q_INVOKABLE void onTestClicked();
Q_INVOKABLE void setValue(int newValue);
Q_INVOKABLE int getValue();
signals:
void valueChanged(int value);
void kkYUy(QString str);
private slots:
void networ_slots();
};
注意:依然需要使用 Q_INVOKABLE 这个宏声明成员函数,将函数申明为元对象系统可调用的函数,Q_INVOKABLE声明后的函数即可在qml中调用。
2.第二步:在main.cpp中进行注册:
#include <QQmlContext>
#include <QQmlEngine>
#include "network.h"
int main(int argc, char *argv[])
{
set_qt_environment();
QGuiApplication app(argc, argv);
qmlRegisterType<NetworkHandler>("NetLibrary", 1, 0, "NetworkHandler");
QQmlApplicationEngine engine;
const QUrl url(u"qrc:/qt/qml/Main/main.qml"_qs);
QObject::connect(
&engine,
&QQmlApplicationEngine::objectCreated,
&app,
[url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
},
Qt::QueuedConnection);
engine.addImportPath(QCoreApplication::applicationDirPath() + "/qml");
engine.addImportPath(":/");
engine.load(url);
if (engine.rootObjects().isEmpty()) {
return -1;
}
return app.exec();
}
注册语句:qmlRegisterType<NetworkHandler>("NetLibrary", 1, 0, "NetworkHandler");
其中NetLibrary是QML中的组件名,1.0是组件版本号,NetworkHandler是用于QML中的类名。
第三步:修改一下QML文件:
import QtQuick 6.0
import QtQuick.Controls 6.0
import NetLibrary 1.0
ApplicationWindow {
visible: true
width: 640
height: 480
title: qsTr("QML Button Example")
Column {
id: buttonRow
anchors.fill: parent
spacing: 20
NetworkHandler{
id:networkClass
}
Button {
id: button1
text: qsTr("button1")
onClicked: {
console.log("Button clicked12!")
networkClass.onConnetClicked()
networkClass.setValue(1)
button1.text = qsTr("Clicked1!")
}
}
}
}
说明:
import NetLibrary 1.0 是在 main.cpp中进行注册了类型的版本化命名空间。
导入之后,就可以如下所示 调用NetworkHandler进行实例化,并设置id为networkClass:
NetworkHandler
{
id:networkClass
}
在两个button的点击事件中,就能通过 实例化对象 networkClass调用 第一步 被 Q_INVOKABLE声明的函数了。
二.方式二:信号槽方式
1.第一步:依然需要新建一个类NetworkHandler:
#include <QObject>
class NetworkHandler : public QObject
{
Q_OBJECT
public:
explicit NetworkHandler(QObject *parent = nullptr);
Q_INVOKABLE void onConnetClicked();
Q_INVOKABLE void onTestClicked();
Q_INVOKABLE void setValue(int newValue);
Q_INVOKABLE int getValue();
signals:
void valueChanged(int value);
void kkYUy(QString str);
private slots:
void networ_slots();
};
注意:信号函数前不用加入Q_INVOKABLE宏
2.第二步:在main.cpp中进行注册:
#include <QQmlContext>
#include <QQmlEngine>
#include "network.h"
int main(int argc, char *argv[])
{
set_qt_environment();
QGuiApplication app(argc, argv);
qmlRegisterType<NetworkHandler>("NetLibrary", 1, 0, "NetworkHandler");
QQmlApplicationEngine engine;
const QUrl url(u"qrc:/qt/qml/Main/main.qml"_qs);
QObject::connect(
&engine,
&QQmlApplicationEngine::objectCreated,
&app,
[url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
},
Qt::QueuedConnection);
engine.addImportPath(QCoreApplication::applicationDirPath() + "/qml");
engine.addImportPath(":/");
engine.load(url);
if (engine.rootObjects().isEmpty()) {
return -1;
}
return app.exec();
}
注册语句:qmlRegisterType<NetworkHandler>("NetLibrary", 1, 0, "NetworkHandler");
其中NetLibrary是QML中的组件名,1.0是组件版本号,NetworkHandler是用于QML中的类名。
第三步:修改一下QML文件:
import QtQuick 6.0
import QtQuick.Controls 6.0
import NetLibrary 1.0
ApplicationWindow {
visible: true
width: 640
height: 480
title: qsTr("QML Button Example")
Column {
id: buttonRow
anchors.fill: parent
spacing: 20
NetworkHandler{
id:networkClass
}
Connections{ //信号-槽连接
target: networkClass
function onValueChanged(value){
console.log("/*-----------收到信号----------*/")
console.log(value)
}
}
Button {
id: button1
text: qsTr("button1")
onClicked: {
console.log("Button clicked12!")
networkClass.onConnetClicked()
networkClass.setValue(1)
button1.text = qsTr("Clicked1!")
}
}
}
}
说明:定义一个槽函数来接收c++中发送过来的信号
Connections{ //信号-槽连接
target: networkClass
function onValueChanged(value){
console.log("/*-----------收到信号----------*/")
console.log(value)
}
}
注意:在c++类中定义的信号是 valueChanged(),在Qml中进行监听 需要在前面加 "on",首字母大写,变为onValueChanged:{}
三.方式三:通过Q_PROPERTY宏来实现
1.第一步:依然需要新建一个类MyClass:
#include <QDebug>
#include <QObject>
class MyClass : public QObject
{
Q_OBJECT
Q_PROPERTY(int value READ getValue WRITE setValue NOTIFY valueChanged)
public:
MyClass(QObject *parent = nullptr) : QObject(parent), m_value(0) {}
int getValue() const {
qDebug()<<"MyClass--------getValue-----:"<<m_value;
return m_value;
}
void setValue(int value) {
if (m_value != value) {
m_value = value;
qDebug()<<"MyClass--------setValue-----:"<<m_value;
emit valueChanged(m_value);
}
}
signals:
void valueChanged(int newValue);
private:
int m_value;
};
说明:
Q_PROPERTY(int value READ getValue WRITE setValue NOTIFY valueChanged) 是 Qt 开发框架中的一个宏定义,用于在 QObject 的派生类中声明一个属性,
并为该属性提供读写接口以及属性值变化时的通知信号。
声明属性:
通过 Q_PROPERTY 宏,可以在 QObject 的派生类中声明一个名为 value 的属性。
该属性可以被 Qt 的属性系统识别和访问。
提供读写接口:
READ getValue 指定了读取 value 属性值的函数 getValue。
WRITE setValue 指定了设置 value 属性值的函数 setValue。
这两个函数分别用于获取和设置属性的值,从而实现了对属性值的封装和保护。
属性变化通知:
NOTIFY valueChanged 指定了当 value 属性值发生变化时发出的通知信号 valueChanged。
这允许其他对象或组件在属性值变化时采取相应的行动,如更新界面或触发其他逻辑。
注意:
这种方式只能修改类的一个属性值,一般类型比较简单,就是int、QString这种;如果需要调用类的自定方法建议还是用方式一中的Q_INVOKABLE宏方式。
2.第二步:在main.cpp中进行注册:
#include "test.h"
int main(int argc, char *argv[])
{
set_qt_environment();
QGuiApplication app(argc, argv);
qmlRegisterType<MyClass>("MyLibrary", 1, 0, "MyClass");
QQmlApplicationEngine engine;
const QUrl url(u"qrc:/qt/qml/Main/main.qml"_qs);
QObject::connect(
&engine,
&QQmlApplicationEngine::objectCreated,
&app,
[url](QObject *obj, const QUrl &objUrl) {
if (!obj && url == objUrl)
QCoreApplication::exit(-1);
},
Qt::QueuedConnection);
engine.addImportPath(QCoreApplication::applicationDirPath() + "/qml");
engine.addImportPath(":/");
engine.load(url);
if (engine.rootObjects().isEmpty()) {
return -1;
}
return app.exec();
}
注册语句:qmlRegisterType<MyClass>("MyLibrary", 1, 0, "MyClass");
第三步:修改一下QML文件:
import QtQuick 6.0
import QtQuick.Controls 6.0
import MyLibrary 1.0
ApplicationWindow {
visible: true
width: 640
height: 480
title: qsTr("QML Button Example")
Column {
id: buttonRow
anchors.fill: parent
spacing: 20
MyClass{
id:myClass
}
Button {
id: button1
text: qsTr("button1")
onClicked: {
console.log("Button clicked12!")
console.log("myClass.value:",myClass.value)
myClass.value=2
button1.text = qsTr("Clicked1!")
}
}
}
}
调用语句:
console.log("myClass.value:",myClass.value) //获取value值
myClass.value=2 //设置value值
运行后log输出:
MyClass--------getValue-----: 0
qml: myClass.value: 0
MyClass--------setValue-----: 2
相关文章:
qml调用c++类内函数的三种方法
一.方法一:使用 Q_INVOKABLE 宏声明成员函数 1.第一步:依然需要新建一个类NetworkHandler: #include <QObject> class NetworkHandler : public QObject { Q_OBJECT public: explicit NetworkHandler(QObject *parent nullptr); Q_INVOKAB…...
NLP任务四大范式的进阶历程:从传统TF-IDF到Prompt-Tuning(提示词微调)
引言:从TF-IDF到Prompt-Tuning(提示词微调),NLP的四次变革 自然语言处理(NLP)技术从最早的手工特征设计到如今的Prompt-Tuning,经历了四个重要阶段。随着技术的不断发展,我们的目标…...
GAMES101:现代计算机图形学入门-笔记-09
久违的101图形学回归咯 今天的话题应该是比较轻松的:聊一聊在渲染中比较先进的topics Advanced Light Transport 首先是介绍一系列比较先进的光线传播方法,有无偏的如BDPT(双向路径追踪),MLT(梅特罗波利斯…...
【Db First】.NET开源 ORM 框架 SqlSugar 系列
.NET开源 ORM 框架 SqlSugar 系列 【开篇】.NET开源 ORM 框架 SqlSugar 系列【入门必看】.NET开源 ORM 框架 SqlSugar 系列【实体配置】.NET开源 ORM 框架 SqlSugar 系列【Db First】.NET开源 ORM 框架 SqlSugar 系列【Code First】.NET开源 ORM 框架 SqlSugar 系列 …...
MySQL聚合查询分组查询联合查询
#对应代码练习 -- 创建考试成绩表 DROP TABLE IF EXISTS exam; CREATE TABLE exam ( id bigint, name VARCHAR(20), chinese DECIMAL(3,1), math DECIMAL(3,1), english DECIMAL(3,1) ); -- 插入测试数据 INSERT INTO exam (id,name, chinese, math, engli…...
告别照相馆!使用AI证件照工具HivisionIDPhotos打造在线证件照制作软件
文章目录 前言1. 安装Docker2. 本地部署HivisionIDPhotos3. 简单使用介绍4. 公网远程访问制作照片4.1 内网穿透工具安装4.2 创建远程连接公网地址 5. 配置固定公网地址 前言 本文主要介绍如何在Linux系统使用Docker快速部署一个AI证件照工具HivisionIDPhotos,并结合…...
通信原理第三次实验
实验目的与内容 实验操作与结果 5.1 刚开始先不加入白噪声,系统设计如下: 正弦波参数设置如下: FM设计如下: 延迟设计如下: 两个滤波器设计参数如下: 输出信号频谱为(未加入噪声)&a…...
【halcon】Metrology工具系列之 get_metrology_object_result_contour
get_metrology_object_result_contour (操作员) 名称 get_metrology_object_result_contour — 查询测量对象的结果轮廓。 签名 get_metrology_object_result_contour( : Contour : MetrologyHandle, Index, Instance, Resolution : ) 描述 get_metrology_object_result_…...
A052-基于SpringBoot的酒店管理系统
🙊作者简介:在校研究生,拥有计算机专业的研究生开发团队,分享技术代码帮助学生学习,独立完成自己的网站项目。 代码可以查看文章末尾⬇️联系方式获取,记得注明来意哦~🌹 赠送计算机毕业设计600…...
NLP信息抽取大总结:三大任务(带Prompt模板)
信息抽取大总结 1.NLP的信息抽取的本质?2.信息抽取三大任务?3.开放域VS限定域4.信息抽取三大范式?范式一:基于自定义规则抽取(2018年前)范式二:基于Bert下游任务建模抽取(2018年后&a…...
python常见问题-pycharm无法导入三方库
1.运行环境 python版本:Python 3.9.6 需导入的greenlet版本:greenlet 3.1.1 2.当前的问题 由于需要使用到greenlet三方库,所以进行了导入,以下是我个人导入时的全过程 ①首先尝试了第1种导入方式:使用pycharm进行…...
迅为RK3588开发板Android系统开发笔记-使用ADB工具
1 使用 ADB 工具 ADB 英文名叫 Android debug bridge ,是 Android SDK 里面的一个工具,用这个工具可以操作管理 Android 模拟器或者真实的 Android 设备,主要的功能如下所示: 在 Android 设备上运行 shell 终端,用命…...
什么是分布式数据库?
随着现代互联网应用和大数据时代的到来,分布式数据库成为了解决大规模数据存储和高并发处理的核心技术之一。本文将通过深入浅出的方式,带你全面理解分布式数据库的概念、工作原理以及底层实现技术。无论你是刚刚接触分布式数据库的开发者,还…...
Leetcode 3363. Find the Maximum Number of Fruits Collected
Leetcode 3363. Find the Maximum Number of Fruits Collected 1. 解题思路2. 代码实现 题目链接:3363. Find the Maximum Number of Fruits Collected 1. 解题思路 这一题是一道陷阱题…… 乍一眼看过去,由于三人的路线完全可能重叠,因此…...
【数据仓库 | Data Warehouse】数据仓库的四大特性
1. 前言 数据仓库是用于支持管理和决策的数据集合,它汇集了来自不同数据源的历史数据,以便进行多维度的分析和报告。数据仓库的四大特点是:主题性,集成性,稳定性,时变性。 2. 主题性(Subject-Oriented) …...
springboot配置多数据源mysql+TDengine保姆级教程
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、pom文件二、yamlDataSourceConfigServiceMapper.xml测试总结 前言 Mybatis-plus管理多数据源,数据库为mysql和TDengine。 一、pom文件 <de…...
dns实验2:反向解析
启动服务: 给虚拟机网卡添加IP地址: 查看有几个IP地址: 打开配置文件: 重启服务,该宽松模式,关闭防火墙: 本机测试: windows测试:(本地shell)...
ZooKeeper 基础知识总结
先赞后看,Java进阶一大半 ZooKeeper 官网这样介绍道:ZooKeeper 是一种集中式服务,用于维护配置信息、命名、提供分布式同步和提供组服务。 各位hao,我是南哥,相信对你通关面试、拿下Offer有所帮助。 ⭐⭐⭐一份南哥编写…...
npm库xss依赖的使用方法和vue3 中Web富文本编辑器 wangeditor 使用xss库解决 XSS 攻击的方法
npm库xss依赖的使用方法和vue3 中Web富文本编辑器 wangeditor 使用xss库解决 XSS 攻击的方法 1. npm库xss依赖的使用方法1.1 xss库定义1.2 xss库功能 2. vue3 中 wangeditor 使用xss库解决 XSS 攻击的方法和示例2.1 在终端执行如下命令安装 xss 依赖2.2 在使用 wangeditor 的地…...
微信小程序蓝牙writeBLECharacteristicValue写入数据返回成功后,实际硬件内信息查询未存储?
问题:连接蓝牙后,调用小程序writeBLECharacteristicValue,返回传输数据成功,查询硬件响应发现没有存储进去? 解决:一直以为是这个write方法的问题,找了很多相关贴,后续进行硬件日志…...
Cursor实现用excel数据填充word模版的方法
cursor主页:https://www.cursor.com/ 任务目标:把excel格式的数据里的单元格,按照某一个固定模版填充到word中 文章目录 注意事项逐步生成程序1. 确定格式2. 调试程序 注意事项 直接给一个excel文件和最终呈现的word文件的示例,…...
盘古信息PCB行业解决方案:以全域场景重构,激活智造新未来
一、破局:PCB行业的时代之问 在数字经济蓬勃发展的浪潮中,PCB(印制电路板)作为 “电子产品之母”,其重要性愈发凸显。随着 5G、人工智能等新兴技术的加速渗透,PCB行业面临着前所未有的挑战与机遇。产品迭代…...
阿里云ACP云计算备考笔记 (5)——弹性伸缩
目录 第一章 概述 第二章 弹性伸缩简介 1、弹性伸缩 2、垂直伸缩 3、优势 4、应用场景 ① 无规律的业务量波动 ② 有规律的业务量波动 ③ 无明显业务量波动 ④ 混合型业务 ⑤ 消息通知 ⑥ 生命周期挂钩 ⑦ 自定义方式 ⑧ 滚的升级 5、使用限制 第三章 主要定义 …...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
线程同步:确保多线程程序的安全与高效!
全文目录: 开篇语前序前言第一部分:线程同步的概念与问题1.1 线程同步的概念1.2 线程同步的问题1.3 线程同步的解决方案 第二部分:synchronized关键字的使用2.1 使用 synchronized修饰方法2.2 使用 synchronized修饰代码块 第三部分ÿ…...
渲染学进阶内容——模型
最近在写模组的时候发现渲染器里面离不开模型的定义,在渲染的第二篇文章中简单的讲解了一下关于模型部分的内容,其实不管是方块还是方块实体,都离不开模型的内容 🧱 一、CubeListBuilder 功能解析 CubeListBuilder 是 Minecraft Java 版模型系统的核心构建器,用于动态创…...
如何将联系人从 iPhone 转移到 Android
从 iPhone 换到 Android 手机时,你可能需要保留重要的数据,例如通讯录。好在,将通讯录从 iPhone 转移到 Android 手机非常简单,你可以从本文中学习 6 种可靠的方法,确保随时保持连接,不错过任何信息。 第 1…...
新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案
随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...
uniapp微信小程序视频实时流+pc端预览方案
方案类型技术实现是否免费优点缺点适用场景延迟范围开发复杂度WebSocket图片帧定时拍照Base64传输✅ 完全免费无需服务器 纯前端实现高延迟高流量 帧率极低个人demo测试 超低频监控500ms-2s⭐⭐RTMP推流TRTC/即构SDK推流❌ 付费方案 (部分有免费额度&#x…...
大学生职业发展与就业创业指导教学评价
这里是引用 作为软工2203/2204班的学生,我们非常感谢您在《大学生职业发展与就业创业指导》课程中的悉心教导。这门课程对我们即将面临实习和就业的工科学生来说至关重要,而您认真负责的教学态度,让课程的每一部分都充满了实用价值。 尤其让我…...
