[TSG开发日志](一)软件基础框架 - 轩先生。
source link: https://www.cnblogs.com/Leventure/p/17677298.html
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.
[TSG开发日志](一)软件基础框架
中间经历了几次波折,最终才算是有时间把软件开发的框架确定下来了。现在开发才终于算是开始有了个起头。
其实在使用Qt做大型软件的时候会遇到一些问题,为此也要不得不做一些妥协。关于这个,我觉得你可能需要看一下这两篇文章
越是深入开发越需要注意一点:Qt只是一个库,一个开发框架,它和boost库,MFC库并没有本质区别,它并不一定能撑起你整个的开发框架,你甚至可能要中途引入很多各种各样的库。
唯一的区别是,Qt算是一个相当重型的库,其中很多内容的生态都是封闭的(当然了,这都是建立在C++狗屎一样的String,如果string的问题可以解决,那么我相信这些库不能兼容的问题都将不复存在),这就意味着如果你使用了Qt的库,你大部分时间都将在这些Q打头的头文件里跌跌撞撞。
做出你想要做的事情,很难走出去或者走进来。这也是为什么大部分情况下用了Qt就不会用别的库了,即使你知道会有这样那样的问题:一是没必要,二是懒得弄。
目前TSG项目开发使用Qt 5.14.2 + vs2019,请按照指定配置来开发,否则出现编译不过的情况请及时更换配置。
本文件将简单介绍一下软件框架及开发要求。框架没什么好说的,软件的要求是:所有新开发的模块都应该继承trunk\TSG_Base\PublicTemplate\TSG_DeviceTemplate.h中给定的指定类型,并在指定应用程序的kernel层中调用其父类,而不关心类的细节。
比如Device类型有很多很多,但是我在应用程序层中并不关心你的Device类型的细节,不管你是Faro,HikCam还是Imu,我只需要用QList将你所有的设备类型给打包,然后只需要管我的Device接口即可。
框架 TSG_Framework
一、底层信号机制 TSG_Caller
请所有的类至少需要继承../Framework/TSG_Framework/目录下的TSG_Framework.h,其中包含了基本的信号和槽,本框架所有消息转发机制都是通过此回调函数进行的:
class TSG_Framework
{
public:
using CallMethod = void(*)(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra);
using SendCMD = void(*)(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra);
void RegisterCallMethod(CallMethod callback) {
callbacks_.append(callback);
}
void RegisterSendCMD(SendCMD callback) {
sendcmds_.append(callback);
}
void Signal_CallMethod(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) {
for (CallMethod callback : callbacks_) {
if (callback) {
callback(sModule, sDescribe, sVariable, extra);
}
}
}
void Signal_SendCMD(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) {
for (SendCMD callback : callbacks_) {
if (callback) {
callback(sModule, sDescribe, sVariable, extra);
}
}
}
virtual qint32 slot_ReturnValue(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
virtual qint32 slot_GetCMD(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
private:
QList<CallMethod> callbacks_;
QList<SendCMD> sendcmds_;
};
在这个父类中,提供了一个最基础的回调函数注册机制和发送控制命令的函数,后续所有继承这个类的子类都需要覆写这个方法,以便能够获得上层的消息。
这里我可能需要做出一些解释:
这就是CallMethod和ReturnValue在上下级中的映射关系,整套系统的控制实际上相当简单,上层控制层(后续包括业务层)通过CallMethod(回调)和ReturnValue方法在Kernel中交换消息,来实现控制层对底层的调用。
也就是说,逻辑上层的设备类如果需要调用底层的功能DLL,比如日志,网络通信等DLL,则需要使用CallMethod方法,如果底层DLL或者是从网络层发来的消息,则需要通过ReturnValue方法向上反应。
这一套信号槽机制构成了这个框架的基本,而上下信息尽量传递Json字符串,而不是一个简单的字符串,这个我们后面设计的时候会提到。
为什么不用信号和槽?因为信号和槽在继承中会出很多问题,而且会导致很多编译问题,这些问题我在做了大量调研后发现是解决不了的,所以只能用回调函数。
除了最基础的类,以下的类大多是建立在TSG_Framework的基石上。
我要先给出其他几个类之间的关系,再来说明这几个类的细节
一个设备的控制大体如上,Interface类实际上只是一个转发和接口,提供给主要进程一个最主要的控制接口,其实主要是信号收发机制。ConfigController更多的是像一个控制类内的Kernel类,但是Device类的成员并不是放在ConfigController类内。
ConfigController是一个真正承上启下的部分,管理设备信息的本地登记、各种信息的获取和修改,都留在本地,然后通过GetCMD获得各种各样的修改,否则都是默认值。这么做是为了保证在没有设置参数的前提下也能够正常执行任务。
当然了,Device并不会是一个单独的,而是会分为两部分,一个是设备本身的控制,第二个是业务的流程控制。原因很简单,设备并不一定是只有一个,而可能是一组设备。
接下来的介绍不分先后,名称也并不会按照上面的来。
二、参数类型声明 TSG_Params
class TSG_Params :public QObject {
public:
QMap<QString, QString> getKeys() {
QMap<QString, QString> ret;
const QMetaObject* metaObject = this->metaObject();
int propertyCount = metaObject->propertyCount();
for (int i = 0; i < propertyCount; ++i)
{
QMetaProperty property = metaObject->property(i);
if (property.isValid() && property.isReadable() && property.isWritable())
{
qDebug() << "Property name: " << property.name();
qDebug() << "Property type: " << property.typeName();
ret.insert(property.typeName(), property.name());
}
}
return ret;
}
QString toJson() {
return Lev_Json::JsonSerialization(this);
}
bool isValid(QString strMessage) {
return Lev_Json::ValidateJsonKeys(strMessage, this);
}
};
实际上对于一个参数类,并没有做什么实际上的操作,因为每一个参数类都是通过元对象去操作这个Object本身,所以这个类可以无限制的拓展。当然了,至于想拓展成什么样还得看在开发中的具体需求。
要求所有在设备控制层中都必须传递参数类,比如
但是在控制层之上的管理层中却无法使用模板类规范输入,这是因为QObject无法支持模板类继承导致的,详情见文章开头的两篇文章
这导致了部分在管理层中只能通过Json字符串来控制控制层中的参数传递,所以必须要对所有的参数传递提供一定的验证手段和转换手段,详情见下图:
另外,在Lev_Json模块中,我提供了一整套的检查方法,可以直接将Json字符串和类内的所有成员变量之间做一个基本的筛查,这也是TSG_Params的基石。没有注释,函数名称就 是注释:
三、设备类声明 TSG_Device
不管设备内部的运行细节,一个设备都应该有以下这些接口:这里不关心设备的具体细节,但是设备本身都必须为这一套工作提供流程,因为这些接口都是在上层中需要被调用的,以这个类为基础去注册具体设备操作。
当然了这样的声明仅限于在框架内部操作,实际上只有业务层提供一个操作接口。
四、设备配置文件控制 TSG_ConfigHelper
ConfigHelper类只需要给定一个路径和指定设备的名称组就可以去自己初始化和获得设备的配置文件了。
class TSG_ConfigHelper : public TSG_Framework {
public:
virtual bool Init(const QString& path, const QList<QString>& list_device_names) = 0;
virtual bool Init(const QString& path) = 0;
virtual QList<QString> getConfigFiles() = 0;
virtual QJsonObject getConfigContain(const QString& config_name, const QString& device_name) = 0;
virtual bool setConfigContain(const QString& config_name, const QString& contains, const QString& device_name) = 0;
virtual QList<QString> getDeviceNames() = 0;
virtual qint32 slot_ReturnValue(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
virtual qint32 slot_GetCMD(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
};
注:对每一个设备来说,其设备的配置文件类型和配置文件模式都应该是已经写死写固定了的,所以只给定一个应用程序当前的运行路径,然后自己去其中检索目录。
五、设备管理类 TSG_Device_Controller
这个就是Interface,没什么好说的,每个模块都是自洽的,so,不用传参了,直接来吧。
class TSG_Device_Controller : public TSG_Framework {
//输入当前进程的路径作为根路径,用于初始化控制类,完全是交由自己控制的
virtual bool Init(const QString& ApplicationPath) = 0;
virtual QString getDeviceType() = 0;
virtual bool PreStartMission() = 0;
virtual bool StartMission() = 0;
virtual bool PauseMission() = 0;
virtual bool EndMission() = 0;
virtual WorkState Get_WorkState() = 0;
qint32 slot_ReturnValue(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
qint32 slot_GetCMD(const QString& sModule, const QString& sDescribe, const QString& sVariable, const QVariant& extra) { return -1; };
};
Recommend
-
42
-
57
日志实现框架 Log4j Log4j 是目前最为流行的 Java日志框架 之一, 1999年 发布首个版本, 2012年 发布最后一个版本, 2015年 正...
-
8
【二】零基础上手HAL库之—熟悉Cubemx软件的框架在完成了Hal库开发环境的搭建后,就是要开始使用此工具了,那么我们肯定有一些必要的准备工作要进行,本节内容仅仅讲解软件各个模块的功能,想要用好软件,第一步当然就是大体上的熟悉软件创建一个工程。...
-
4
《PhotoP 开发日志 03》- 基础体验完善
-
3
上期文章我们讲到了ArkUI的三大特性,同时提到了ArkUI是一套用于构建HarmonyOS应用界面的UI开发框架,本期我们将从架构设计上来聊聊ArkUI的设计理念。
-
5
一.ACE_Engine框架概述: ACE_Engine框架是OpenHarmony 的UI开发框架,提供开发者在进行应用UI开发时所必需的各种组件,以及定义这些组件的属性,样式,事件及方法。通过这些组件可以很方便的进行OpenHarmony上UI应用的开发。 ACE_Engine提供...
-
3
【大型软件开发】开发日志(五).net框架与C++的融合:CLR——C++调用C#的DLL ...
-
6
🐊🐊🐊🐊🐊好多小鳄鱼 一、关于串口通信: Qt的确有自己的串口通信类,就是QSerialPort,但是我们在使用过程中因为要更加定制化的使用串口通信类减小开发的难度,所以我们会提供一个串口通信类,也就是这个SerialPortHelper类。...
-
5
OpenHarmony AI框架开发指导 1、 功能简介 AI业务子系统是OpenHarmony提供原生的分布式AI能力的子系统。AI业务子系统提供了统一的AI引擎框架,实现算法能力快速插件化集成。 AI引擎框架主要包含插件管理、模块管理和...
-
7
并发编程防御装-锁(基础版) 大家好,我是小高先生。在Java并发编程的世界中,锁的地位至关重要。它就像是一道坚固的防线,确保了并发编程运行结果的正确性。你可以不准备攻击装备,但是锁这个防御装备是必不可少的。相信大...
About Joyk
Aggregate valuable and interesting links.
Joyk means Joy of geeK