以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「englyf」https://mp.weixin.qq.com/s/o-_aGqreuQda-ZmKktvxwA


以往在公司开发众多的项目中,都会出现要求本项目里部分功能模块代码需要具备保密性。如果需要对外输出demo工程,那么需要做到不会泄密。

举一下爪子,以前做雷达开发的时候,客户从公司那儿买了这些雷达模块,也会需要从作为供应商的厂家这里拿到这些模块的开发demo(也就是示范软件工程)。但是由于demo里可能会用到一些涉及计算的内部算法函数调用,这些函数的实现属于公司商业机密是不能对外公开的,于是就提出了对部分代码保密的需求。

那么,怎么引入和管理这些需要保密的代码模块呢?一般的做法就是将需要保密的功能模块以子项目的形式引入到demo工程,这些子项目在编译时输出库文件,而demo工程则通过子项目提供的接口头文件引用这些库提供的接口。最后在打包输出demo工程之前,会把涉及保密功能模块所属的子项目源码从工程目录树中删除掉。被清理掉的子项目只留下生成的库文件和引用头文件(如果你的工程用的开发语言是C/C++)。

下面就继续用 Qt Quick 的开发示例来展开(果然。。。),这个示例里会实现一个简单的加法GUI小工具,计算过程所用到的算法以库API的形式提供,算法源码封装在库里。

本示例工程完整代码见文末链接分享!

设计工程目录

为了方便调用涉密的功能模块的接口以及打包输出不包含涉密模块源码的demo工程,接口的实现源码存放在对应模块的子工程(子项目)目录内,而接口的声明所在的头文件和涉密模块编译后输出的库文件需要存放在对应模块子项目的文件夹之外。之所以这样安排存放位置,是为了方便打包输出时对demo工程的清理。

看看我的工程目录:

这里有两个项目,一个顶层工程项目,也就是demo工程,命名为 gettingStartWithQtQuickSubProject,存放目录与工程名一致。另一个是子项目,里边代表着涉密模块的功能源码,命名为math。每个项目都对应一个CMakeLists.txt配置文件,其他文件视乎项目内容需要而定。

配置子项目

子项目 ./math/src/ 目录下存放着涉密模块的实现源码,这里只是简单地演示了加法的计算接口

/* inside file MyMath.cpp */
#include "MyMath.h" myMath::myMath(QObject *parent) : QObject(parent)
{} QVariant myMath::sum(QVariant a, QVariant b)
{
return a.toInt() + b.toInt();
}

定义接口源文件

子项目math的所有源文件都存放在 ./math/src/ 目录下,所以将该目录指定为源文件搜索目录,并将搜索结果存到cmake自定义变量DIR_SRCS中,

aux_source_directory(${CMAKE_CURRENT_SOURCE_DIR}/src DIR_SRCS)

变量CMAKE_CURRENT_SOURCE_DIR是cmake自带变量,指向当前所属项目的配置文件所在目录,也即是 ./math/。

包含接口头文件

接口声明所在的头文件存放在demo项目的./src/include/中,在子项目中包含对应的目录需要通过demo项目的源码目录相对地指定。

include_directories(${CMAKE_SOURCE_DIR}/src/include)

变量CMAKE_SOURCE_DIR指向的内容就是顶层项目的配置文件(CMakeLists.txt)所在目录,这里的demo项目就是顶层项目。

配置输出库文件

子项目编译后输出的库文件应该自动输出到demo项目的./src/libs/中,可以通过设置cmake变量LIBRARY_OUTPUT_PATH来实现。

SET(LIBRARY_OUTPUT_PATH "${CMAKE_SOURCE_DIR}/src/libs")

配置顶层项目(demo项目)

我们以demo项目为顶层项目,根据实际项目需要,本来是可以添加各式各样的子项目,但是这里根据示例需要仅添加math子项目。

添加子项目

本项目作为顶层项目,配置文件(CMakeLists.txt)中除了要描述清楚本项目的配置内容,还需要指明添加了哪些子项目,而且子项目需要在编译顶层项目之前被编译完成。

add_subdirectory(${CMAKE_CURRENT_SOURCE_DIR}/math)

如上,利用指令add_subdirectory添加了子项目math。此语句必须在add_executable语句之前。在完整工程编译链接后,打包输出demo工程时,需要手动删除此语句,因为客户在拿到demo工程后,不需要重新编译子项目的内容,而是直接使用库文件即可。

添加接口头文件

为了简化添加源文件和头文件的过程,采用递归目录搜索的方式,将指定目录下所有需要的文件检索到cmake自定义变量SRC_FILES中,方便后边的编译引用

file(GLOB_RECURSE SRC_FILES
${CMAKE_CURRENT_SOURCE_DIR}/src/*.cpp
${CMAKE_CURRENT_SOURCE_DIR}/src/*.h
)

接口头文件所在的目录./src/include也包含在上面的检索过程中了

添加链接库文件

指定链接目录以及链接的库math

link_directories(${CMAKE_CURRENT_SOURCE_DIR}/src/libs)
link_libraries(math)

输出demo工程包

完整编译执行一遍demo工程后,库文件会被自动保存在demo工程目录下。编译过程中,各项目是分开按顺序编译链接的,先是子项目,然后才是顶层项目demo。

为了之后能单独编译链接demo项目内容,需要删除math目录,并且在demo工程的顶层CMakeLists.txt文件中,把添加子项目math的语句删除即可。

执行一下上面的工程

点击=执行

好了,简单演示到这里,欢迎后续交流...


本示例工程完整代码git仓库链接: git@github.com:ifi-leung/gettingStartWithQtQuickSubProject.git

Qt Quick 用cmake怎么玩子项目的更多相关文章

  1. 一文入门Qt Quick

    以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「englyf」https://www.cnblogs.com/englyf/p/16733091.html 初识Qt Quick 很高兴可以 ...

  2. Qt Quick实现的疯狂算数游戏

    使用 Qt Quick 写了个小游戏:疯狂算数.支持 Windows 和 Android 两个平台. 游戏简单,但牵涉到下面你的 Qt Quick 主题: 自己实现一个按钮 自适应分辨率 国际化 QM ...

  3. Qt Widgets、QML、Qt Quick的区别

    Qt Widgets.QML.Qt Quick的区别 简述 看了之前关于 QML 的一些介绍,很多人难免会有一些疑惑: Q1:QML 和 Qt Quick 之间有什么区别? Q2:QtQuick 1. ...

  4. 浏览器上的Qt Quick

    你想不想在浏览器上运行你的Qt Quick程序呢?在Qt 5.12之前,唯一的方法是使用Qt WebGL Streaming技术把界面镜像到浏览器上.但该方法有不少缺陷,下文会说.前不久随着Qt 5. ...

  5. 《Qt Quick 4小时入门》学习笔记4

    http://edu.csdn.net/course/detail/1042/14806?auto_start=1 Qt Quick 4小时入门 第七章:处理鼠标与键盘事件 1.处理鼠标事件 鼠标信号 ...

  6. 《Qt Quick 4小时入门》学习笔记3

    http://edu.csdn.net/course/detail/1042/14807?auto_start=1 Qt Quick 4小时入门 第八章:Qt Quick中的锚(anchors)布局 ...

  7. 《Qt Quick 4小时入门》学习笔记2

    http://edu.csdn.net/course/detail/1042/14805?auto_start=1   Qt Quick 4小时入门 第五章:Qt Quick基本界面元素介绍   1. ...

  8. 《Qt Quick 4小时入门》学习笔记

    http://edu.csdn.net/course/detail/1042/14804?auto_start=1   Qt Quick 4小时入门 第五章:Qt Quick里的信号与槽   QML中 ...

  9. 从头学Qt Quick(3)-- 用QML写一个简单的颜色选择器

    先看一下效果图: 实现功能:点击不同的色块可以改变文字的颜色. 实现步骤: 一.创建一个默认的Qt Quick工程: 二.添加文件Cell.qml 这一步主要是为了实现一个自定义的组件,这个组件就是我 ...

随机推荐

  1. vue2与vue3实现响应式的原理区别和提升

    区别: vue2.x: 实现原理: 对象类型:Object.defineProperty()对属性的读取,修改进行拦截(数据劫持): 数组类型:通过重写更新数组的一系列方法来进行拦截(对数组的变更方法 ...

  2. NC20471 [ZJOI2007]棋盘制作

    题目链接 题目 题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名. 据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑白相间的方阵,对应八八六十四卦,黑白对 ...

  3. 在微信小程序中,如何获取 for 循环的 index

    微信小程序中,for 循环的 index(索引值)可以用wx:for-index="index"来获取. <view class="item" wx:fo ...

  4. 向日葵远程RCE漏洞分析及漏洞利用脚本编写

    0x00 漏洞概述 向日葵是一款免费的,集远程控制电脑.手机.远程桌面连接.远程开机.远程管理.支持内网穿透等功能的一体化远程控制管理软件.如果想要手机远控电脑,或者电脑远控手机可以利用向日葵:如果是 ...

  5. 2020牛客NOIP赛前集训营-提高组(第三场) C - 牛半仙的妹子Tree (树链剖分)

    昨天教练问我:你用树剖做这道题,怎么全部清空状态呢?    我:???不是懒标记就完了???    教练:树剖不是要建很多棵线段树吗,不止log个,你要一个一个清?    我:为什么要建很多棵线段树? ...

  6. ZJU-199001 第三周练习 2 数字特征值 位运算算法

    题目 对数字求特征值是常用的编码算法,奇偶特征是一种简单的特征值. 对于一个整数, 从个位开始对每一位数字编号, 个位是 \(1\) 号, 十位是 \(2\) 号, 以此类推. 这个整数在第位上的数字 ...

  7. 【HTML】学习路径3-段落标签和标题标签

    第一章:标题标签 <h1>  </h1> <h2>  </h2> 等等... 数字越大,字体尺寸越小. <!--标题会加粗.独占一行--> ...

  8. CF-1562B- Scenes From a Memory

    Problem - 1562B - Codeforces 题意:给定一个字符串,每次操作可以选择这个字符串中的一种字符,将他们全部都减1,最多K次操作,问可以形成的字典大小最小的字符串. 题解:首先我 ...

  9. VS Code 之KoroFileHeader插件

    设置 在vscode左下角点击设置按钮,选择"设置",然后输入"fileheader", 文件头部注释:Fileheader:custom Made 函数注释: ...

  10. 《!--suppress ALL --> 在Android XML 文件中的用途是什么?

    <!--suppress ALL --> 在Android XML 文件中的用途是什么? 警告一次又一次地出现在谷歌地图的 XML 文件中,但是当我使用时,所有警告都被禁用.那么压制所有评 ...