cmake引用静态或动态库(四)

前言

上一篇文章讲解了如何构建静态库或者动态库,本文目标为:
1、引入外部静态库
2、引入外部动态库

准备工作

创建sample7,在里面创建3rdlilb,将上一篇文章构建的Math.h头文件、以及将lib文件夹下内容全部拷贝进来,创建src源码目录,将sample2的main.cpp拷贝进来,最终目录如下:

./sample7
    |
    +--- CMakeLists.txt
    |
    +--- build/
    +--- src/
            +--- CMakeLists.txt
            +--- main.cpp
    +--- lib/
          +--- libMath.a
          +--- libmath.1.2.dylib
          +--- libmath.1.dylib
          +--- libmath.dylib
          +--- Math.h

这里要再main.cpp中应用外部math库,其内容为:

#include 
#include 
#include "Math.h"

int main(int argc, char *argv[])
{
    if (argc < 3){
        printf("Usage: %s base exponent \n", argv[0]);
        return 1;
    }
    double base = atof(argv[1]);
    int exponent = atoi(argv[2]);
    double result = power(base, exponent);
    printf("%g ^ %d is %g\n", base, exponent, result);
    return 0;
}

工程根目录下CMakeLists.txt内容为:

# CMake 最低版本号要求
cmake_minimum_required (VERSION 2.8)

if(POLICY CMP0042)
  cmake_policy(SET CMP0042 NEW)  # CMake 3.0+ (2.8.12): MacOS "@rpath" in target's install name
endif()

# 项目工程名
project (sample7)
message(STATUS "root This is BINARY dir " ${PROJECT_BINARY_DIR})
message(STATUS "root This is SOURCE dir " ${PROJECT_SOURCE_DIR})

# 添加子目录
ADD_SUBDIRECTORY(src)

1、引入外部静态库

src目录下的CMakeLists.txt文件内容为:

# 打印信息
message(STATUS "src This is BINARY dir " ${PROJECT_BINARY_DIR})
message(STATUS "src This is SOURCE dir " ${PROJECT_SOURCE_DIR})

# 定义工程根目录; CMAKE_SOURCE_DIR为内建变量,表示工程根目录的CMakeLists.txt文件路径
SET(ROOT_DIR ${CMAKE_SOURCE_DIR})

# 指定头文件搜索路径
INCLUDE_DIRECTORIES(${ROOT_DIR}/3rdlib)

# 指定引用的外部库的搜索路径
LINK_DIRECTORIES(${ROOT_DIR}/3rdlib)

# 指定可执行文件存放目录
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)

# 构建可执行程序
ADD_EXECUTABLE(sample7 main.cpp)
TARGET_LINK_LIBRARIES(sample7 libmath.a)
  • INCLUDE_DIRECTORIES(${ROOT_DIR}/3rdlib)
    用于指定头文件的搜索路径,因为Math.h与main.cpp不在同一目录,所以如果不配置这个搜索路径,肯定会报错
  • LINK_DIRECTORIES(${ROOT_DIR}/3rdlib)
    指定静态库或者动态库的搜索路径
  • TARGET_LINK_LIBRARIES(sample7 libmath.a)
    指定要连接的静态库,必须要有,第二个参数也可以换成math,那么将自动去搜索libmath.a或者libmath.so或者libmath.dyld动态库

还有另外一种写法,如下:

# 打印信息
message(STATUS "src This is BINARY dir " ${PROJECT_BINARY_DIR})
message(STATUS "src This is SOURCE dir " ${PROJECT_SOURCE_DIR})

# 定义工程根目录; CMAKE_SOURCE_DIR为内建变量,表示工程根目录的CMakeLists.txt文件路径
SET(ROOT_DIR ${CMAKE_SOURCE_DIR})

# 指定头文件搜索路径
INCLUDE_DIRECTORIES(${ROOT_DIR}/3rdlib)

# 以导入外部库的方式(不会重新编译)重新构建一个静态库libMath
add_library(libMath STATIC IMPORTED)
# 设置要导入的外部静态库的路径
set_target_properties(libMath PROPERTIES IMPORTED_LOCATION ${ROOT_DIR}/3rdlib/libmath.a)

# 指定可执行文件存放目录
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)

# 构建可执行程序
ADD_EXECUTABLE(sample7 main.cpp)
# 注意这里连接的库名为新构建的库名libMath而外部库libmath
TARGET_LINK_LIBRARIES(sample7 libMath)

这里解释下,这里是先根据外部静态库重新构建一个libMath静态库,然后可执行程序连接到该库

  • add_library(libMath STATIC IMPORTED)
    以导入外部库的方式重新构建一个静态库libMath,不会重新编译
  • set_target_properties(libMath PROPERTIES IMPORTED_LOCATION ${ROOT_DIR}/3rdlib/libmath.a)

设置要导入的外部静态库的路径

tips:
建议用方式一导入外部库

2、引入外部动态库

修改src下CMakeLists.txt文件内容:

# 打印信息
message(STATUS "src This is BINARY dir " ${PROJECT_BINARY_DIR})
message(STATUS "src This is SOURCE dir " ${PROJECT_SOURCE_DIR})

# 定义工程根目录; CMAKE_SOURCE_DIR为内建变量,表示工程根目录的CMakeLists.txt文件路径
SET(ROOT_DIR ${CMAKE_SOURCE_DIR})

# 指定头文件搜索路径
INCLUDE_DIRECTORIES(${ROOT_DIR}/3rdlib)

# 指定引用的外部库的搜索路径
LINK_DIRECTORIES(${ROOT_DIR}/3rdlib)

# 指定可执行文件存放目录
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)

# 构建可执行程序
ADD_EXECUTABLE(sample7 main.cpp)
TARGET_LINK_LIBRARIES(sample7 libmath.dylib)

tips:
因为sample7 在构建的时候引入了动态库,所以当./sample7 在执行的时候将去指定的目录加载这个动态库(这个路径非编译时指定的动态库路径,发现会报错"dyld: Library not loaded: @rpath/libmath.1.dylib") ,如果将他们拷贝到/usr/local/lib/目录下,发现就不报错了。对于mac 系统,它默认按照如下顺序搜索动态库:

1.编译目标代码时指定的动态库搜索路径(通过 -Wl,-rpath参数指定的)
2.环境变量LD_LIBRARY_PATH指定的动态库搜索路径;
3.配置文件/etc/ld.so.conf中指定的动态库搜索路径;
4.默认的动态库搜索路径/usr/lib
5.默认的动态库搜索路径/usr/local/lib

其它系统可能有些区别,但大体都差不多

引入静态库和动态库的区别

1、引入静态库时,静态库在连接阶段会被连接到最终目标中(比如可执行执行程序中),缺点就是同一份静态库如果被不同的程序引用,那么内存中会存在这个静态库函数的多份拷贝
2、引入动态库时,连接阶段不会被拷贝最终目标中,程序运行时将按照指定的规则(上一步提到的规则,并非编译时的动态库路径)去搜索这个动态库,搜索到了之后才加载到内存中。所以多个程序就算引用了同一个动态库,内存中也只是存在一份动态库函数的拷贝

CMake 环境变量关键字

1、CMAKE_INCLUDE_PATH
2、CMAKE_LIBRARY_PATH
这两个变量在cmake语法中显示调用并没有效果,它的作用给CMake的FIND__XXX()系列函数提供查找路径,比如如下代码

# 在环境变量CMAKE_LIBRARY_PATH指定的目录中查找libmath.dylib
# 的路径,并赋值给mypath变量
message(STATUS "ddd This is SOURCE dir " ${CMAKE_INCLUDE_PATH})
FIND_PATH(myHeader hello.h)
FIND_LIBRARY(mypath libmath.dylib)
message(STATUS "ee This is SOURCE dir " ${myHeader})
message(STATUS "ttt This is SOURCE dir " ${mypath})

将libmath.dyld,Math.h分别拷贝到/usr/local/lib,/usr/local/include目录中,执行

export PATH=$PATH:/usr/local/lib:/usr/local/include
cmake ..

发现${myHeader}和${mypath}都是有值的

你可能感兴趣的:(cmake引用静态或动态库(四))