以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「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. PostgreSQL定时备份

    # PostgreSQL定时备份 # 版本:PostgreSQL14.0 # 一.如果你在pg_hba.conf中设定了密码登录,那么你就需要设置一个默认密码共psql登录的时候跳过验证.如果是roo ...

  2. Nginx 集群部署(Keepalived)

    # Nginx集群部署 # 当我们的用户同时访问量达到一定量的时候,一台服务器是不够用的 # 这个时候我们需要解决这个问题肯定是要添加新的服务器去处理用户访问 # 多台服务器处理用户访问就需要我们集群 ...

  3. AI识万物:从0搭建和部署手语识别系统 ⛵

    作者:韩信子@ShowMeAI 深度学习实战系列:https://www.showmeai.tech/tutorials/42 计算机视觉实战系列: https://www.showmeai.tech ...

  4. Apache DolphinScheduler 如何从 1.2.1 升级到 1.3.4

    关于 Apache DolphinScheduler Apache DolphinScheduler(incubator) 于 17 年在易观数科立项, 19 年 8 月进入 Apache 孵化器,已 ...

  5. Linux 09 Vim

    参考源 https://www.bilibili.com/video/BV187411y7hF?spm_id_from=333.999.0.0 版本 本文章基于 CentOS 7.6 概述 Vi Vi ...

  6. mongo数据库-mongo角色权限

    Built-In Roles(内置角色): 1. 数据库用户角色:read.readWrite; 2. 数据库管理角色:dbAdmin.dbOwner.userAdmin: 3. 集群管理角色:clu ...

  7. linux下用docker安装redis

    docker安装redis方法: 1.用命令来查看可用版本: docker search redis 2.拉取官方的最新版本的镜像:docker pull redis:latest 3.查看镜像:do ...

  8. [RootersCTF2019]I_<3_Flask-1|SSTI注入

    1.打开之后很明显的提示flask框架,但是未提供参数,在源代码中发现了一个git地址,打开之后也是没有啥用,结果如下: 2.这里我们首先需要爆破出来参数进行信息的传递,就需要使用Arjun这一款工具 ...

  9. sftp服务器的搭建与连接

    在个人租用的vps上搭建sftp服务器并通过本机连接上去 本实验所需:Xshell(xshell中包含xftp).一台vps(windows系统) sftp是一种协议,即SSH File Transf ...

  10. 强大多云混合多K8S集群管理平台Rancher入门实战

    @ 目录 概述 定义 为何使用 其他产品 安装 简述 规划 基础环境 Docker安装 Rancher安装 创建用户 创建集群 添加Node节点 配置kubectl 创建项目和名称空间 发布应用 偏好 ...