find_package()
原理解读根据cmake官方文档可以知道,find_package()
有Module模式(基本用法,basic signature)和Config模式(full signature,完全用法),其中Module模式是基础,Config模式则更复杂高级些。
Module模式也就是基础用法(Basic Signature,这里Signature表示“用法”,而不是“签名”),Config模式也就是高级用法(Full Signature)。
The
CONFIG
option, the synonymousNO_MODULE
option, or the use of options not specified in the basic signature all enforce pure Config mode. In pure Config mode, the command skips Module mode search and proceeds at once with Config mode search.
也就是说,只有这3种情况下才是Config模式:
find_package()
中指定CONFIG
关键字find_package()
中指定NO_MODULE
关键字find_package()
中使用了不在"basic signature"(也就是Module模式下所有支持的配置)关键字换句话说,只要我不指定"CONFIG",不指定“NO_MODULE",也不使用"full signature"中的关键字,那我就是在Module模式。排查find_package()
的第一步,应当判断它是Module模式还是Config模式。
find-package.jpg
image.png
find_package()
的用法find_package(
[version] [EXACT] [QUIET] [MODULE]
[REQUIRED] [[COMPONENTS] [components...]]
[OPTIONAL_COMPONENTS components...]
[NO_POLICY_SCOPE])
Module模式下,相比于Config模式,可选配置参数少一些,并且如果按用户指定的配置却找不到包,就会自动进入Config模式(如上图所示)。
关键字解释version
和EXACT
: 都是可选的,version
指定的是版本,如果指定就必须检查找到的包的版本是否和version
兼容。如果指定EXACT
则表示必须完全匹配的版本而不是兼容版本就可以。
QUIET
可选字段,表示如果查找失败,不会在屏幕进行输出(但是如果指定了REQUIRED
字段,则QUIET
无效,仍然会输出查找失败提示语)。
MODULE
可选字段。前面提到说“如果Module模式查找失败则回退到Config模式进行查找”,但是假如设定了MODULE
选项,那么就只在Module模式查找,如果Module模式下查找失败并不回落到Config模式查找。
REQUIRED
可选字段。表示一定要找到包,找不到的话就立即停掉整个cmake。而如果不指定REQUIRED
则cmake会继续执行。
COMPONENTS
,components
:可选字段,表示查找的包中必须要找到的组件(components),如果有任何一个找不到就算失败,类似于REQUIRED
,导致cmake停止执行。
OPTIONAL_COMPONENTS
和components
:可选的模块,找不到也不会让cmake停止执行。
Module模式查找顺序
Module模式下是要查找到名为Find
的文件。
先在CMAKE_MODULE_PATH
变量对应的路径中查找。如果路径为空,或者路径中查找失败,则在cmake module directory(cmake安装时的Modules目录,比如/usr/local/share/cmake/Modules
)查找。
find_package()
的用法find_package(
[version] [EXACT] [QUIET]
[REQUIRED] [[COMPONENTS] [components...]]
[CONFIG|NO_MODULE]
[NO_POLICY_SCOPE]
[NAMES name1 [name2 ...]]
[CONFIGS config1 [config2 ...]]
[HINTS path1 [path2 ... ]]
[PATHS path1 [path2 ... ]]
[PATH_SUFFIXES suffix1 [suffix2 ...]]
[NO_DEFAULT_PATH]
[NO_PACKAGE_ROOT_PATH]
[NO_CMAKE_PATH]
[NO_CMAKE_ENVIRONMENT_PATH]
[NO_SYSTEM_ENVIRONMENT_PATH]
[NO_CMAKE_PACKAGE_REGISTRY]
[NO_CMAKE_BUILDS_PATH] # Deprecated; does nothing.
[NO_CMAKE_SYSTEM_PATH]
[NO_CMAKE_SYSTEM_PACKAGE_REGISTRY]
[CMAKE_FIND_ROOT_PATH_BOTH |
ONLY_CMAKE_FIND_ROOT_PATH |
NO_CMAKE_FIND_ROOT_PATH])
Config模式下的查找顺序,比Module模式下要多得多。而且,新版本的CMake比老版本的有更多的查找顺序(新增的在最优先的查找顺序)。它要找的文件名字也不一样,Config模式要找
或
。查找顺序为:
_ROOT
的cmake变量或环境变量。CMake3.12新增。设定CMP0074 Policy来关闭。_DIR
cmake变量,那么_ROOT
不起作用。举例:
cmake_minimum_required(VERSION 3.13)
project(fk_cmk)
set(OpenCV_ROOT "F:/zhangzhuo/lib/opencv_249/build")
set(OpenCV_DIR "F:/zhangzhuo/lib/opencv_300/build")
find_package(OpenCV QUIET
NO_MODULE
NO_DEFAULT_PATH
NO_CMAKE_PATH
NO_CMAKE_ENVIRONMENT_PATH
NO_SYSTEM_ENVIRONMENT_PATH
NO_CMAKE_PACKAGE_REGISTRY
NO_CMAKE_BUILDS_PATH
NO_CMAKE_SYSTEM_PATH
NO_CMAKE_SYSTEM_PACKAGE_REGISTRY
)
message(STATUS "OpenCV library status:")
message(STATUS " version: ${OpenCV_VERSION}")
message(STATUS " libraries: ${OpenCV_LIBS}")
message(STATUS " include path: ${OpenCV_INCLUDE_DIRS}")
实际上会找到opencv300,也就是OpenCV_DIR
这一cmake变量的值最先起作用。
CMAKE_PREFIX_PATH
CMAKE_FRAMEWORK_PATH
CMAKE_APPBUNDLE_PATH
可以通过设定NO_CMAKE_PATH
来关闭这一查找顺序
_DIR
CMAKE_PREFIX_PATH
CMAKE_FRAMEWORK_PATH
CMAKE_APPBUNDLE_PATH
可以通过NO_CMAKE_ENVIRONMENT_PATH
来跳过。
HINT
字段指定的路径
搜索标准的系统环境变量PATH。
其中如果是以/bin
或者/sbin
结尾的,会自动转化为其父目录。
通过指定NO_SYSTEM_ENVIRONMENT_PATH
来跳过。
存储在cmake的"User Package Registry"(用户包注册表)中的路径。
通过设定NO_CMAKE_PACKAGE_REGISTRY
,或者:
设定CMAKE_FIND_PACKAGE_NO_PACKAGE_REGISTRY
为true,
来避开。
设定为当前系统定义的cmake变量:
CMAKE_SYSTEM_PREFIX_PATH
CMAKE_SYSTEM_FRAMEWORK_PATH
CMAKE_SYSTEM_APPBUNDLE_PATH
通过设定NO_CMAKE_SYSTEM_PATH
来跳过。
在cmake的"System Package Registry"(系统包注册表)中查找。
通过设定NO_CMAKE_SYSTEM_PACKAGE_REGISTRY
跳过。
或者通过设定CMAKE_FIND_PACKAGE_NO_SYSTEM_PACKAGE_REGISTRY
为true。
从PATHS
字段指定的路径中查找。