四、官方示例 directconnectdynamicclient (simpleSwitch)

2024-06-20 07:38

本文主要是介绍四、官方示例 directconnectdynamicclient (simpleSwitch),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1 将replica 添加到项目中
  • 2 创建远程节点并将其连接到源主机节点。
  • 3 获取远程源对象的副本
  • 4 需要注意的点
    • 1 只有当Replica发出initialized信号后,该Replica才有Source端的元信息(属性、信号与槽),才能被使用。
    • 2 静态 Vs. 动态
      • 静态方式的优劣:

动态连接方式实现QtRo

在源端不需要做任何更改,因为动态副本只会影响请求者节点获取副本的方式。因此,我们使用上一章节中所示的源代码。

1 将replica 添加到项目中

因为replica 是动态获取的,所以不需要rep文件。

2 创建远程节点并将其连接到源主机节点。

此步骤的代码与示例1 directconnectclient 中的代码相同

  QRemoteObjectNode repNode; // create remote object noderepNode.connectToNode(QUrl(QStringLiteral("local:switch"))); // connect with remote host node

3 获取远程源对象的副本

#include <QCoreApplication>#include "dynamicclient.h"int main(int argc, char *argv[]){QCoreApplication a(argc, argv);QSharedPointer<QRemoteObjectDynamicReplica> ptr; // shared pointer to hold replicaQRemoteObjectNode repNode;repNode.connectToNode(QUrl(QStringLiteral("local:switch")));ptr.reset(repNode.acquireDynamic("SimpleSwitch")); // acquire replica of source from host nodeDynamicClient rswitch(ptr); // create client switch object and pass replica reference to it}

dynamicclient.h

#ifndef _DYNAMICCLIENT_H#define _DYNAMICCLIENT_H#include <QObject>#include <QSharedPointer>#include <QRemoteObjectNode>#include <qremoteobjectdynamicreplica.h>class DynamicClient : public QObject{Q_OBJECTpublic:DynamicClient(QSharedPointer<QRemoteObjectDynamicReplica> ptr);~DynamicClient();Q_SIGNALS:void echoSwitchState(bool switchState);// this signal is connected with server_slot(..) slot of source object and echoes back switch state received from sourcepublic Q_SLOTS:void recSwitchState_slot(); // Slot to receive source statevoid initConnection_slot(); //Slot to connect signals/slot on replica initializationprivate:bool clientSwitchState; // holds received server switch stateQSharedPointer<QRemoteObjectDynamicReplica> reptr;// holds reference to replica};#endif

dynamicclient.cpp

 #include "dynamicclient.h"// constructorDynamicClient::DynamicClient(QSharedPointer<QRemoteObjectDynamicReplica> ptr) :QObject(nullptr), reptr(ptr){//connect signal for replica valid changed with signal slot initializationQObject::connect(reptr.data(), SIGNAL(initialized()), this, SLOT(initConnection_slot()));}//destructorDynamicClient::~DynamicClient(){}// Function to initialize connections between slots and signalsvoid DynamicClient::initConnection_slot(){// connect source replica signal currStateChanged() with client's recSwitchState() slot to receive source's current stateQObject::connect(reptr.data(), SIGNAL(currStateChanged()), this, SLOT(recSwitchState_slot()));// connect client's echoSwitchState(..) signal with replica's server_slot(..) to echo back received stateQObject::connect(this, SIGNAL(echoSwitchState(bool)),reptr.data(), SLOT(server_slot(bool)));}void DynamicClient::recSwitchState_slot(){clientSwitchState = reptr->property("currState").toBool(); // use replica property to get currState from sourceqDebug() << "Received source state " << clientSwitchState;Q_EMIT echoSwitchState(clientSwitchState); // Emit signal to echo received state back to server}

4 需要注意的点

1 只有当Replica发出initialized信号后,该Replica才有Source端的元信息(属性、信号与槽),才能被使用。

即我们需要

 QObject::connect(reptr.data(), SIGNAL(initialized()), this, SLOT(initConnection_slot()));

收到initialized信号后,才初始化我们的副本端代码(需要用到source信息)

2 静态 Vs. 动态

有了静态和动态两种方式后我们该如何选择呢?在这里我根据自己的实践经验给出一个比较:

静态方式的优劣:

优:

  • 拥有明确的定义,更适合在C++中使用(因为有repc生成的头文件)。 支持POD等复杂结构的定义。
  • 更高效。因为结构定义都已经在C++中定义好了,不需要动态传输、构建,省去了这方面的开销(其实不大,但多少是有开销的)。

缺:

  • Source端和Replica端必须严格使用同一版本的rep文件,即使rep文件内只是添加了一行注释,否则连接不上。这个是我们当时用静态方式最大的痛苦。我们一般将大系统分成多个模块,各模块开发测试通常都是并行的。静态的这个缺点导致我们在测试时经常连接不上。

动态方式的优劣:
优:

  • Source端和Replica端不再需要严格使用同一版本的rep文件,因为Replica端已经不需要该文件了,所以Source端随时更改(当然不能改接口定义,不然Replica端肯定没法调用)。
  • Source端可以对接口进行版本管理,在兼容旧接口的情况下,动态Replica不需要做任何更新。这个其实是和上面这条联系在一起的。
  • 对于QML编程来说使用方法和静态方式是一样的。

缺:

  • 在C++端使用起来非常不便。因为没有了repc生成的头文件,所有信号与槽等元信息必须通过Qt的元编程来实现,比较繁琐。
  • 不支持POD等复杂结构定义。但我也讲了,可以用QVariantList和QVariantMap两兄弟来曲线救国。
  • 必须等初始化后才能使用,给编程增加了额外复杂度,同时增加了构建连接的额外开销。这个是动态这个特性决定的。

这篇关于四、官方示例 directconnectdynamicclient (simpleSwitch)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1077511

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

ModelMapper基本使用和常见场景示例详解

《ModelMapper基本使用和常见场景示例详解》ModelMapper是Java对象映射库,支持自动映射、自定义规则、集合转换及高级配置(如匹配策略、转换器),可集成SpringBoot,减少样板... 目录1. 添加依赖2. 基本用法示例:简单对象映射3. 自定义映射规则4. 集合映射5. 高级配置匹

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

Java实现自定义table宽高的示例代码

《Java实现自定义table宽高的示例代码》在桌面应用、管理系统乃至报表工具中,表格(JTable)作为最常用的数据展示组件,不仅承载对数据的增删改查,还需要配合布局与视觉需求,而JavaSwing... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码

C++ 检测文件大小和文件传输的方法示例详解

《C++检测文件大小和文件传输的方法示例详解》文章介绍了在C/C++中获取文件大小的三种方法,推荐使用stat()函数,并详细说明了如何设计一次性发送压缩包的结构体及传输流程,包含CRC校验和自动解... 目录检测文件的大小✅ 方法一:使用 stat() 函数(推荐)✅ 用法示例:✅ 方法二:使用 fsee

mysql查询使用_rowid虚拟列的示例

《mysql查询使用_rowid虚拟列的示例》MySQL中,_rowid是InnoDB虚拟列,用于无主键表的行ID查询,若存在主键或唯一列,则指向其,否则使用隐藏ID(不稳定),推荐使用ROW_NUM... 目录1. 基本查询(适用于没有主键的表)2. 检查表是否支持 _rowid3. 注意事项4. 最佳实