Linux 下cmake生成SO库的使用实例

本文详细介绍了如何使用CMake构建一个C++共享库,并将其链接到一个可执行项目中。通过具体步骤和代码示例,展示了从创建共享库到在项目中使用的过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、创建我的共享库:MySharedLib
CMakeLists.txt

    cmake_minimum_required(VERSION 2.8)
    project(MySharedLib)
    # C++11 编译
    set(CMAKE_CXX_STANDARD 11)
    #创建共享库
    add_library(MySharedLib SHARED library.cpp library.h)

library.h

  

  #ifndef MYSHAREDLIB_LIBRARY_H
    #define MYSHAREDLIB_LIBRARY_H
     
    // 打印 Hello World!
    void hello();
     
    // 使用可变模版参数求和
    template <typename T>
    T sum(T t)
    {
        return t;
    }
    template <typename T, typename ...Types>
    T sum(T first, Types ... rest)
    {
        return first + sum<T>(rest...);
    }
     
    #endif

library.cpp

    #include <iostream>
    #include "library.h"
     
    void hello() {
        std::cout << "Hello, World!" << std::endl;
    }

在CMakeLists.txt所在的目录中:

cmake  ./
make

so 共享库的使用(被可执行项目调用)创建一个名为test的可执行项目
CMakeLists.txt

   

 #cmake版本
    cmake_minimum_required(VERSION 2.8)
     
    #项目名:test
    project(test)
     
    # C++11 编译
    set(CMAKE_CXX_STANDARD 11)
     
    #指定生成的版本
    #set(CMAKE_BUILD_TYPE DEBUG)
     
    #指定编译选项
    #set(CMAKE_CXX_FLAGS_DEBUG "-g -Wall")
     
    # 指定头文件路径
    set(INC_DIR /home/share/MySharedLib)
     
    # 库文件路径 指定静态和动态文件目录
    set(LIB_DIR /home/share/MySharedLib)
     
    #指定头文件目录
    include_directories(${INC_DIR})
     
    #链接库路径
    link_directories(${LIB_DIR})
     
    #链接库,MySharedLib为库名
    link_libraries(MySharedLib)
     
    #生成目标文件
    add_executable(test main.cpp)
    #指定源代码
    #set(SOURCE_FILES main.cpp)
    #add_executable(BoostCoroutineDemo ${SOURCE_FILES})
     
    # 链接 库文件 MySharedLib 库
    target_link_libraries(test MySharedLib)
    #target_link_libraries(BoostCoroutineDemo libboost_system.a libboost_thread.a)

main.cpp

 

   #include <iostream>
    #include "library.h"
    using std::cout;
    using std::endl;
     
    int main() {
     
        hello();
        cout << "1 + 2 = " << sum(1,2) << endl;
        cout << "1 + 2 + 3 = " << sum(1,2,3) << endl;
     
        return 0;
    }

cd  testMylib/
cmake ./
make
./test

---------------------
作者:cindywry
来源:优快云
原文:https://blog.youkuaiyun.com/cindywry/article/details/86063930
版权声明:本文为博主原创文章,转载请附上博文链接!

<think>好的,用户的问题是:在Linux下如何使用CMake将共享(.so)转换为静态(.a)。我需要结合提供的引用信息来解答这个问题。 首先,我需要回忆CMake的基本知识。CMake通过CMakeLists.txt文件来配置项目构建过程,控制生成静态或共享。根据引用[2],CMake可以简化构建流程,替代Makefile的复杂性。 然后,用户想从生成共享改为静态。通常在CMake中,使用add_library命令来指定的类型。例如,add_library(mylib SHARED src.cpp)生成.so,而改为STATIC则生成.a。因此,用户需要修改CMakeLists.txt中的add_library指令,将SHARED改为STATIC。引用[3]中的例子提到了target_link_libraries,但这里的关键是的类型定义。 不过,可能还存在依赖问题。如果项目中依赖的其他是动态,转换成静态可能需要调整链接方式。比如,如果原来的共享依赖其他动态,改为静态后可能需要将这些依赖也静态链接。引用[4]提到了动态加载的问题,但这里可能不直接相关。 另外,用户可能需要确保所有依赖项都有静态版本可用。例如,如果使用第三方,需要有对应的.a文件而非.so。否则,链接时可能会出错。引用[5]提到交叉编译libyuv,可能需要配置正确的编译选项来生成静态。 还有一个考虑是CMake的BUILD_SHARED_LIBS变量。如果用户希望全局控制的类型,可以设置set(BUILD_SHARED_LIBS OFF),这样默认生成静态,而不需要修改每个add_library命令。 总结步骤: 1. 修改CMakeLists.txt中的add_library,将SHARED改为STATIC。 2. 确保依赖有静态版本,或调整链接方式。 3. 可能设置BUILD_SHARED_LIBS为OFF。 4. 重新运行CMake生成构建文件,并编译。 相关问题方面,可以涉及如何同时生成两种类型、静态链接依赖项的问题、使用CMake变量控制类型,以及处理静态中的符号冲突等。例如,引用[4]中的段错误可能与动态加载选项有关,但静态可能避免此类问题。</think>要在Linux环境下使用CMake将共享(.so)转换为静态(.a),需通过修改构建配置实现。以下是具体步骤及原理说明: --- ### **步骤1:修改`CMakeLists.txt`文件** 在`CMakeLists.txt`中找到定义共享的代码段: ```cmake add_library(mylib SHARED src1.cpp src2.cpp) ``` 将`SHARED`改为`STATIC`: ```cmake add_library(mylib STATIC src1.cpp src2.cpp) # 生成静态mylib.a ``` --- ### **步骤2:处理依赖** 如果项目依赖其他(如OpenCV),需确保链接时使用静态: ```cmake target_link_libraries(mylib ${OpenCV_LIBS} # 假设OpenCV_LIBS已指向静态路径 ) ``` 若第三方仅提供动态,需重新编译其静态版本[^2]。 --- ### **步骤3:全局配置(可选)** 通过设置`BUILD_SHARED_LIBS`变量强制所有默认静态编译: ```cmake set(BUILD_SHARED_LIBS OFF) # 放在CMakeLists.txt开头 ``` --- ### **步骤4:重新构建项目** ```bash mkdir build && cd build cmake .. # 生成Makefile make # 编译生成mylib.a ``` --- ### **关键问题说明** 1. **符号冲突**:静态会直接嵌入目标程序,需避免同名符号重复定义[^4]。 2. **依赖兼容性**:若依赖通过`dlopen`动态加载(如插件系统),需调整代码逻辑。 3. **编译优化**:静态可通过`-fPIC`参数支持位置无关代码,便于后续链接到其他[^5]。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值