热门标签 | 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)

这样就达到我们的预期了


推荐阅读
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
  • 本文详细介绍如何使用arm-eabi-gdb调试Android平台上的C/C++程序。通过具体步骤和实用技巧,帮助开发者更高效地进行调试工作。 ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 在Ubuntu 16.04 LTS上配置Qt Creator开发环境
    本文详细介绍了如何在Ubuntu 16.04 LTS系统中安装和配置Qt Creator,涵盖了从下载到安装的全过程,并提供了常见问题的解决方案。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 本题通过将每个矩形视为一个节点,根据其相对位置构建拓扑图,并利用深度优先搜索(DFS)或状态压缩动态规划(DP)求解最小涂色次数。本文详细解析了该问题的建模思路与算法实现。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文详细分析了Hive在启动过程中遇到的权限拒绝错误,并提供了多种解决方案,包括调整文件权限、用户组设置以及环境变量配置等。 ... [详细]
  • 本文详细介绍了如何使用ActionScript 3.0 (AS3) 连接并操作MySQL数据库。通过具体的代码示例和步骤说明,帮助开发者理解并实现这一过程。 ... [详细]
  • 使用Python在SAE上开发新浪微博应用的初步探索
    最近重新审视了新浪云平台(SAE)提供的服务,发现其已支持Python开发。本文将详细介绍如何利用Django框架构建一个简单的新浪微博应用,并分享开发过程中的关键步骤。 ... [详细]
  • 本文详细介绍了美国最具影响力的十大财团,包括洛克菲勒、摩根、花旗银行等。这些财团在历史发展过程中逐渐形成,并对美国的经济、政治和社会产生深远影响。 ... [详细]
  • Hadoop入门与核心组件详解
    本文详细介绍了Hadoop的基础知识及其核心组件,包括HDFS、MapReduce和YARN。通过本文,读者可以全面了解Hadoop的生态系统及应用场景。 ... [详细]
  • Composer Registry Manager:PHP的源切换管理工具
    本文介绍了一个用于Composer的源切换管理工具——Composer Registry Manager。该项目旨在简化Composer包源的管理和切换,避免与常见的CRM系统混淆,并提供了详细的安装和使用指南。 ... [详细]
  • 本题探讨如何通过最大流算法解决农场排水系统的设计问题。题目要求计算从水源点到汇合点的最大水流速率,使用经典的EK(Edmonds-Karp)和Dinic算法进行求解。 ... [详细]
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社区 版权所有