返回介绍

3.8 检测 Boost 库

发布于 2025-05-06 21:45:55 字数 5664 浏览 0 评论 0 收藏

NOTE : 此示例代码可以在 https://github.com/dev-cafe/cmake-cookbook/tree/v1.0/chapter-03/recipe-08 中找到,包含一个 C++的示例。该示例在 CMake 3.5 版(或更高版本) 中是有效的,并且已经在 GNU/Linux、macOS 和 Windows 上进行过测试。

Boost 是一组 C++通用库。这些库提供了许多功能,这些功能在现代 C++项目中不可或缺,但是还不能通过 C++标准使用这些功能。例如,Boost 为元编程、处理可选参数和文件系统操作等提供了相应的组件。这些库中有许多特性后来被 C++11、C++14 和 C++17 标准所采用,但是对于保持与旧编译器兼容性的代码库来说,许多 Boost 组件仍然是首选。

本示例将向您展示如何检测和链接 Boost 库的一些组件。

准备工作

我们将编译的源码是 Boost 提供的文件系统库与文件系统交互的示例。这个库可以跨平台使用,并将操作系统和文件系统之间的差异抽象为一致的 API。下面的代码( path-info.cpp ) 将接受一个路径作为参数,并将其组件的报告打印到屏幕上:

#include <iostream>
​
#include <boost/filesystem.hpp>
​
using namespace std;
using namespace boost::filesystem;
const char *say_what(bool b) { return b ? "true" : "false"; }
int main(int argc, char *argv[])
{
  if (argc < 2)
  {
    cout
        << "Usage: path_info path-element [path-element...]\n"
           "Composes a path via operator/= from one or more path-element arguments\n"
           "Example: path_info foo/bar baz\n"
#ifdef BOOST_POSIX_API
           " would report info about the composed path foo/bar/baz\n";
#else // BOOST_WINDOWS_API
           " would report info about the composed path foo/bar\\baz\n";
#endif
    return 1;
  }
  path p;
  for (; argc > 1; --argc, ++argv)
    p /= argv[1]; // compose path p from the command line arguments
  cout << "\ncomposed path:\n";
  cout << " operator<<()---------: " << p << "\n";
  cout << " make_preferred()-----: " << p.make_preferred() << "\n";
  cout << "\nelements:\n";
  for (auto element : p)
    cout << " " << element << '\n';
  cout << "\nobservers, native format:" << endl;
#ifdef BOOST_POSIX_API
  cout << " native()-------------: " << p.native() << endl;
  cout << " c_str()--------------: " << p.c_str() << endl;
#else // BOOST_WINDOWS_API
  wcout << L" native()-------------: " << p.native() << endl;
  wcout << L" c_str()--------------: " << p.c_str() << endl;
#endif
  cout << " string()-------------: " << p.string() << endl;
  wcout << L" wstring()------------: " << p.wstring() << endl;
  cout << "\nobservers, generic format:\n";
  cout << " generic_string()-----: " << p.generic_string() << endl;
  wcout << L" generic_wstring()----: " << p.generic_wstring() << endl;
  cout << "\ndecomposition:\n";
  cout << " root_name()----------: " << p.root_name() << '\n';
  cout << " root_directory()-----: " << p.root_directory() << '\n';
  cout << " root_path()----------: " << p.root_path() << '\n';
  cout << " relative_path()------: " << p.relative_path() << '\n';
  cout << " parent_path()--------: " << p.parent_path() << '\n';
  cout << " filename()-----------: " << p.filename() << '\n';
  cout << " stem()---------------: " << p.stem() << '\n';
  cout << " extension()----------: " << p.extension() << '\n';
  cout << "\nquery:\n";
  cout << " empty()--------------: " << say_what(p.empty()) << '\n';
  cout << " is_absolute()--------: " << say_what(p.is_absolute()) << '\n';
  cout << " has_root_name()------: " << say_what(p.has_root_name()) << '\n';
  cout << " has_root_directory()-: " << say_what(p.has_root_directory()) << '\n';
  cout << " has_root_path()------: " << say_what(p.has_root_path()) << '\n';
  cout << " has_relative_path()--: " << say_what(p.has_relative_path()) << '\n';
  cout << " has_parent_path()----: " << say_what(p.has_parent_path()) << '\n';
  cout << " has_filename()-------: " << say_what(p.has_filename()) << '\n';
  cout << " has_stem()-----------: " << say_what(p.has_stem()) << '\n';
  cout << " has_extension()------: " << say_what(p.has_extension()) << '\n';
  return 0;
}

具体实施

Boost 由许多不同的库组成,这些库可以独立使用。CMake 可将这个库集合,表示为组件的集合。 FindBoost.cmake 模块不仅可以搜索库集合的完整安装,还可以搜索集合中的特定组件及其依赖项(如果有的话)。我们将逐步建立相应的 CMakeLists.txt :

  1. 首先,声明 CMake 最低版本、项目名称、语言,并使用 C++11 标准:
    cmake_minimum_required(VERSION 3.5 FATAL_ERROR)
    ​
    project(recipe-08 LANGUAGES CXX)
    ​
    set(CMAKE_CXX_STANDARD 11)
    set(CMAKE_CXX_EXTENSIONS OFF)
    set(CMAKE_CXX_STANDARD_REQUIRED ON)
  2. 然后,使用 find_package 搜索 Boost。若需要对 Boost 强制性依赖,需要一个参数。这个例子中,只需要文件系统组件,所以将它作为参数传递给 find_package :
    find_package(Boost 1.54 REQUIRED COMPONENTS filesystem)
  3. 添加可执行目标,编译源文件:
    add_executable(path-info path-info.cpp)
  4. 最后,将目标链接到 Boost 库组件。由于依赖项声明为 PUBLIC ,依赖于 Boost 的目标将自动获取依赖项:
    target_link_libraries(path-info
      PUBLIC
          Boost::filesystem
        )

工作原理

FindBoost.cmake 是本示例中所使用的 CMake 模块,其会在标准系统安装目录中找到 Boost 库。由于我们链接的是 Boost::filesystem ,CMake 将自动设置包含目录并调整编译和链接标志。如果 Boost 库安装在非标准位置,可以在配置时使用 BOOST_ROOT 变量传递 Boost 安装的根目录,以便让 CMake 搜索非标准路径:

$ cmake -D BOOST_ROOT=/custom/boost

或者,可以同时传递包含头文件的 BOOST_INCLUDEDIR 变量和库目录的 BOOST_LIBRARYDIR 变量:

$ cmake -D BOOST_INCLUDEDIR=/custom/boost/include -DBOOST_LIBRARYDIR=/custom/boost/lib

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。