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

本文介绍如何使用CMake创建共享库,并将其链接到可执行项目中。通过示例展示了CMakeLists.txt的配置,包括设置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

<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]。 ---
评论 5
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值