以下内容为本人的著作,如需要转载,请声明原文链接微信公众号「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. 游戏开发常遇到数据一致性BUG,怎么解?

    摘要:数据副本强一致.全节点可写.存储全面降本,GaussDB(for Redis)重新定义游戏数据库,彻底修复一致性BUG. 本文分享自华为云社区<华为云GaussDB(for Redis)揭 ...

  2. redis-string类型命令

    一.APPEND key value 如果 key 已经存在,并且值为字符串,那么这个命令会把 value 追加到原来值(value)的结尾. 如果 key 不存在,那么它将首先创建一个空字符串的ke ...

  3. laravel框架中验证后在页面提示错误信息

    {{-- 显示错误信息 判断:如果有错误则进行显示,--}} {{-- 通过$errors->any() 获取是否有错误,如果有则返回布尔值true,没有返回布尔值false--}} @if($ ...

  4. sql报错收集

    踩坑记录: 当出现如下错误 Not enough parameters for the SQL statement 多半是因为插入时填写的字段名有误 json.decoder.JSONDecodeEr ...

  5. 使用man手册查看内核函数

    1.为避免man的一些功能缺失,先装好manpages-dev sudo apt-get install manpages-dev 2.安装mandocs的依赖包xmlto sudo apt-get ...

  6. 【pytest官方文档】解读- 如何自定义mark标记,并将测试用例的数据传递给fixture函数

    在之前的分享中,我们知道可以使用yield或者return关键字把fixture函数里的值传递给test函数. 这种方法很实用,比如我在fixture函数里向数据库里插入必要的测试数据,那我就可以把相 ...

  7. spark 解决 java.util.Date is not a valid external type for schema of Date

    出错伪代码如下: //出错的点在这里 import java.util.Date ... val t_rdd = t_frame.rdd.map(row => { val photo_url = ...

  8. csrf跨站请求伪造、csrf相关装饰器、auth认证模块、基于django中间件设计项目功能

    目录 csrf跨站请求网站 什么是csrf跨站请求网站 经典例子-钓鱼网站 模拟 如何避免这种现象(预防) 如何在django中解决这个问题 form表单 ajax csrf相关装饰器 FBV CBV ...

  9. Gitea 与 Jenkins 的集成实践,打造你的专属 CI/CD 系统

    前言 Gitea 是一个用于代码托管的轻量级单体程序,它能与现有的经典应用集成,诸如代码分析工具 SonarQube.持续集成工具 Drone.Jenkins 以及用于工单管理的客户端插件(VSCod ...

  10. 《Java基础——break与continue用法详解》

    Java基础--break与continue用法详解       1. break语句: 规则: 1. 仅用于循环语句和switch语句当中,用于跳出循环. 2. 当只有一层循环时,则直接跳出循环,不 ...