热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

cmake编译带版本的动态库以及链接不带版本的库

个人博客https:juejin.cnuser176366088104638和http:blog.wuzhenyu.com.cncmake编译动态库和链接动态库cmake中&#x

个人博客 https://juejin.cn/user/176366088104638 和 http://blog.wuzhenyu.com.cn



cmake 编译动态库和链接动态库

cmake 中,通过 add_library 的方式,来设置编译目标,编译结果为动态库或者静态库

add_library( [STATIC | SHARED | MODULE][EXCLUDE_FROM_ALL][source1] [source2 ...])

name 就是目标名,即 target_name。目标名称,在 cmake 中是一个很特殊的存在,特殊在哪里呢,后面我一点点展开说明。

上面的参数中,STATIC 表示目标为静态库,而 SHARED 表示为动态库。

我们来看一个例子,例子很简单,就是实现一个 output 打印接口,编译成动态库 liboutput.so,然后通过链接这个动态库的方式调用 output 方法,打印 Hello World 到屏幕上,我们来看一下目录结构

├── CMakeLists.txt
├── demo
│   ├── CMakeLists.txt
│   └── helloworld.cpp
├── output.cpp
├── output.h

根目录中的 CMakeLists.txt 文件为

project(test)add_library(output SHARED output.cpp)
set(LIB_OUTPUT_DIR "${PROJECT_SOURCE_DIR}/dist")
set_target_properties(outputPROPERTIESLIBRARY_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR}ARCHIVE_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR})add_subdirectory(demo)

为了方便,我们通过 set_target_properties 将动态库编译后,存放到根目录下的 dist 文件夹中,${PROJECT_SOURCE_DIR} 这个变量所代表的目录,跟 project 有关,表示的是指定了 project 的目录作为源代码路径,也就是 ${PROJECT_SOURCE_DIR} 这个变量的值。

demo/CMakeLists.txt

cmake_minimum_required(VERSION 2.8)include_directories(${PROJECT_SOURCE_DIR})
add_executable(helloworld helloworld.cpp)
target_link_libraries(helloworld PUBLIC output)

编译成可执行文件 helloworld,编译成功后,看下链接的情况

# ldd helloworld linux-vdso.so.1 (0x00007ffc93ff9000)liboutput.so => /home/jona/test/dist/liboutput.so (0x00007f1ba8c16000)libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f1ba888d000)libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f1ba8675000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f1ba8284000)libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f1ba7ee6000)/lib64/ld-linux-x86-64.so.2 (0x00007f1ba901b000)

在 dist 目录下的动态库也编译成功了,看下目录情况

├── CMakeLists.txt
├── demo
│   ├── CMakeLists.txt
│   └── helloworld.cpp
├── dist
│   ├── liboutput.so
├── output.cpp
├── output.h

可执行程序 helloworld 已经成功链接到了 liboutput.so 这个动态库上。

一般来说,我们编译动态库的时候,都会加上版本号,比如 liboutput.so.0.0.1 ,然后可执行文件在链接的时候,链接到 liboutput.so,让 liboutput.soliboutput.so.0.0.1 的软链接即可。这样,我们在升级不同版本的动态库的时候,只需要修改软链接执行不同版本的动态库即可,不需要重新编译链接源程序。

但是,笔者在使用 cmake 的时候,就遇到了一些坑。我通过下面这种方式来编译的动态库,根目录下的 CMakeLists.txt 改为

project(test)add_library(output SHARED output.cpp)file(STRINGS "VERSION" LIB_VERSION)set(LIB_OUTPUT_DIR "${PROJECT_SOURCE_DIR}/dist")set_target_properties(outputPROPERTIESVERSION ${LIB_VERSION}SOVERSION ${LIB_VERSION}LIBRARY_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR}ARCHIVE_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR})add_subdirectory(demo)

set_target_properties 中,加上了 SOVERSION 版本号,这样,在编译的时候,就会编译成带有版本号的动态库文件,然后创建一个不带有版本号的软链接,变成完成的库如下所示

lrwxrwxrwx 1 jona jona 18 Apr 23 18:19 liboutput.so -> liboutput.so.0.0.1
-rwxrwxr-x 1 jona jona 8648 Apr 23 18:19 liboutput.so.0.0.1

再看下可在执行文件的情况

# ldd helloworld linux-vdso.so.1 (0x00007ffdedbfc000)liboutput.so.0.0.1 => /home/jona/test/dist/liboutput.so.0.0.1 (0x00007f5e480d2000)libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f5e47d49000)libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f5e47b31000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f5e47740000)libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f5e473a2000)/lib64/ld-linux-x86-64.so.2 (0x00007f5e484d7000)

链接的是 liboutput.so.0.0.1 ,为什么不是链接的 liboutput.so 呢,虽然我们在 CMakeLists.txt 文件中是

target_link_libraries(helloworld PUBLIC output)

这样指定的动态库。这里的 output 是目标名,也就是上面我们通过

add_library(output SHARED output.cpp)

指定的目标名,cmake 在链接的时候,通过这个 target name 找到这个库的完整路径进行链接,而库的名称就是liboutput.so.0.0.1,而不是 liboutput.so。我们可以看一下 cmake 生成的链接文件 link.txt

/usr/bin/c++ -rdynamic CMakeFiles/helloworld.dir/helloworld.cpp.o -o helloworld -Wl,-rpath,/home/jona/Documents/programming/c_plus_plus/cmakefile_test/test/dist ../../dist/liboutput.so.0.0.1

链接的时候,直接使用的就是带有版本号的库文件名。

注意,这里即使按照下面这种方式进行链接

/usr/bin/c++ -rdynamic CMakeFiles/helloworld.dir/helloworld.cpp.o -o helloworld -Wl,-rpath,/home/jona/Documents/programming/c_plus_plus/cmakefile_test/test/dist ../../dist/liboutput.so

用 ldd 查看结果,仍然链接的是带版本好的库,因为不带版本号的库,就是一个软链接,实际的库就是带有版本号的。


target_link_libraries 中指定链接库的方式有如下这几种

  • A library target name,就是上面我们使用到的
  • A full path to a library file,这是指定库的完整路径的方式
  • A plain library name,这种方式比较特殊,cmake 会将这种方式翻译成 -lname 或者 name.lib 的方式

比如,我们将上面的改成 target_link_libraries(helloworld PUBLIC output.so) 的方式,link.txt 就变成了

/usr/bin/c++ -rdynamic CMakeFiles/helloworld.dir/helloworld.cpp.o -o helloworld -Wl,-rpath,/home/jona/Documents/programming/c_plus_plus/cmakefile_test/test/dist -loutput

  • A link flag,这种方式,在名称前面加上 -,就变成了 linker 的选项了

那么,我们如何才能做到预期的那样,直接链接到不带版本号的库呢,借助一点小技巧。在编译成动态库的时候,不加版本号,在编译结束后,将库重命名成带有版本号的库,然后创建库的软链接为不带版本号的库,CMakeLists.txt 文件改成如下的方式

project(test)add_library(output SHARED output.cpp)file(STRINGS "VERSION" LIB_VERSION)set(LIB_OUTPUT_DIR "${PROJECT_SOURCE_DIR}/dist")set_target_properties(outputPROPERTIESLIBRARY_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR}ARCHIVE_OUTPUT_DIRECTORY ${LIB_OUTPUT_DIR})add_custom_command(TARGET output POST_BUILDCOMMANDmv liboutput.so liboutput.so.${LIB_VERSION}COMMANDln -s liboutput.so.${LIB_VERSION} liboutput.soWORKING_DIRECTORY ${LIB_OUTPUT_DIR})add_subdirectory(demo)

再看下可执行文件的链接情况

# ldd helloworld linux-vdso.so.1 (0x00007fff453c9000)liboutput.so => /home/jona/test/dist/liboutput.so (0x00007f6209f38000)libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007f6209baf000)libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007f6209997000)libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f62095a6000)libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007f6209208000)/lib64/ld-linux-x86-64.so.2 (0x00007f620a33d000)

这样就达到我们的预期了


推荐阅读
  • CentOS 7.6环境下Prometheus与Grafana的集成部署指南
    本文旨在提供一套详细的步骤,指导读者如何在CentOS 7.6操作系统上成功安装和配置Prometheus 2.17.1及Grafana 6.7.2-1,实现高效的数据监控与可视化。 ... [详细]
  • 本文深入探讨了UNIX/Linux系统中的进程间通信(IPC)机制,包括消息传递、同步和共享内存等。详细介绍了管道(Pipe)、有名管道(FIFO)、Posix和System V消息队列、互斥锁与条件变量、读写锁、信号量以及共享内存的使用方法和应用场景。 ... [详细]
  • 2017-2018年度《网络编程与安全》第五次实验报告
    本报告详细记录了2017-2018学年《网络编程与安全》课程第五次实验的具体内容、实验过程、遇到的问题及解决方案。 ... [详细]
  • cJinja:C++编写的轻量级HTML模板引擎
    本文介绍了cJinja,这是一个用C++编写的轻量级HTML模板解析库。它利用ejson来处理模板中的数据替换(即上下文),其语法与Django Jinja非常相似,功能强大且易于学习。 ... [详细]
  • 本文详细探讨了Java命令行参数的概念、使用方法及在实际编程中的应用,包括如何通过命令行传递参数给Java程序,以及如何在Java程序中解析这些参数。 ... [详细]
  • 本文将指导如何向ReactJS计算器应用添加必要的功能,使其能够响应用户操作并正确计算数学表达式。 ... [详细]
  • 掌握Mosek矩阵运算,轻松应对优化挑战
    本篇文章继续深入探讨Mosek学习笔记系列,特别是矩阵运算部分,这对于优化问题的解决至关重要。通过本文,您将了解到如何高效地使用Mosek进行矩阵初始化、线性代数运算及约束域的设定。 ... [详细]
  • Python3 中使用 lxml 模块解析 XPath 数据详解
    XPath 是一种用于在 XML 文档中查找信息的路径语言,同样适用于 HTML 文件的搜索。本文将详细介绍如何利用 Python 的 lxml 模块通过 XPath 技术高效地解析和抓取网页数据。 ... [详细]
  • 在寻找轻量级Ruby Web框架的过程中,您可能会遇到Sinatra和Ramaze。两者都以简洁、轻便著称,但它们之间存在一些关键区别。本文将探讨这些差异,并提供详细的分析,帮助您做出最佳选择。 ... [详细]
  • KMP算法是处理字符串匹配的一种高效算法它首先用O(m)的时间对模板进行预处理,然后用O(n)的时间完成匹配。从渐进的意义上说,这样时间复 ... [详细]
  • 解决Windows下创建子进程时代码重复执行的问题
    在Windows系统中,当启动子进程时,主进程的文件会被复制到子进程中。由于导入模块时会执行该模块中的代码,因此可能导致某些代码在主进程和子进程中各执行一次。本文探讨了这一现象的原因及其解决方案。 ... [详细]
  • java文本编辑器,java文本编辑器设计思路
    java文本编辑器,java文本编辑器设计思路 ... [详细]
  • 本文深入探讨了 PHP 实现计划任务的方法,包括其原理、具体实现方式以及在不同操作系统中的应用。通过详细示例和代码片段,帮助开发者理解和掌握如何高效地设置和管理定时任务。 ... [详细]
  • 本文深入探讨了 Delphi 中类对象成员的核心概念,包括 System 单元的基础知识、TObject 类的定义及其方法、TClass 的作用以及对象的消息处理机制。文章不仅解释了这些概念的基本原理,还提供了丰富的补充和专业解答,帮助读者全面理解 Delphi 的面向对象编程。 ... [详细]
  • 本文详细介绍了Java中的注解功能,包括如何定义注解类型、设置注解的应用范围及生命周期,并通过具体示例展示了如何利用反射机制访问注解信息。 ... [详细]
author-avatar
以真做假公馆
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有