生成 .so 和使用 .so

是哟ing两个最简单的项目演示使用 c++ 项目生成 .so 和在 c++ 项目中使用 .so

项目1:生成 .so 文件 (maker_so)

项目结构

复制代码
maker_so/
├── include/
│   └── calculator.h
├── src/
│   └── calculator.cpp
└── CMakeLists.txt

文件内容

include/calculator.h

cpp 复制代码
#ifndef CALCULATOR_H
#define CALCULATOR_H

extern "C" {
    int add(int a, int b);
    int subtract(int a, int b);
    int multiply(int a, int b);
    float divide(int a, int b);
}

#endif

src/calculator.cpp

cpp 复制代码
#include "calculator.h"

int add(int a, int b) {
    return a + b;
}

int subtract(int a, int b) {
    return a - b;
}

int multiply(int a, int b) {
    return a * b;
}

float divide(int a, int b) {
    if (b == 0) return 0;
    return (float)a / b;
}

CMakeLists.txt

cmake 复制代码
cmake_minimum_required(VERSION 3.10)
project(maker_so)

# 设置库的输出目录为项目下的 lib
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/lib)

# 包含头文件目录
include_directories(${CMAKE_SOURCE_DIR}/include)

# 创建动态库
add_library(calculator SHARED src/calculator.cpp)

message(STATUS "Building calculator shared library")

构建项目1

bash 复制代码
# 创建并进入项目目录
mkdir -p maker_so/{include,src,build}
cd maker_so

# 创建文件(使用上面的内容)
# ... 创建 calculator.h, calculator.cpp, CMakeLists.txt

# 构建
cd build
cmake ..
cmake --build .

# 查看生成的 .so
ls -l ../lib/
# 应该看到: libcalculator.so

项目2:使用 .so 文件 (user_so)

项目结构

复制代码
user_so/
├── lib/                    # 从项目1复制过来的 .so
│   └── libcalculator.so
├── include/                # 从项目1复制过来的头文件
│   └── calculator.h
├── src/
│   └── main.cpp
└── CMakeLists.txt

文件内容

include/calculator.h(从项目1复制)

cpp 复制代码
#ifndef CALCULATOR_H
#define CALCULATOR_H

extern "C" {
    int add(int a, int b);
    int subtract(int a, int b);
    int multiply(int a, int b);
    float divide(int a, int b);
}

#endif

src/main.cpp

cpp 复制代码
#include <iostream>
#include "calculator.h"

int main() {
    int a = 15, b = 4;
    
    std::cout << "=== Using calculator library ===\n";
    std::cout << a << " + " << b << " = " << add(a, b) << std::endl;
    std::cout << a << " - " << b << " = " << subtract(a, b) << std::endl;
    std::cout << a << " * " << b << " = " << multiply(a, b) << std::endl;
    std::cout << a << " / " << b << " = " << divide(a, b) << std::endl;
    
    return 0;
}

CMakeLists.txt

cmake 复制代码
cmake_minimum_required(VERSION 3.10)
project(user_so)

# 设置可执行文件输出目录
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/bin)

# 设置 RPATH,让程序能在 lib 目录找到 .so
set(CMAKE_INSTALL_RPATH "${CMAKE_SOURCE_DIR}/lib")
set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE)

# 包含头文件目录
include_directories(${CMAKE_SOURCE_DIR}/include)

# 指定库文件目录
link_directories(${CMAKE_SOURCE_DIR}/lib)

# 创建可执行文件
add_executable(calc_app src/main.cpp)

# 链接动态库
target_link_libraries(calc_app calculator)

message(STATUS "Building application that uses calculator.so")

完整的一键构建脚本

创建一个脚本 build_all.sh 来自动化整个过程:

bash 复制代码
#!/bin/bash

echo "=== 项目1: 生成 libcalculator.so ==="

# 创建项目1
mkdir -p maker_so/{include,src,build}

# 创建项目1的文件
cat > maker_so/include/calculator.h << 'EOF'
#ifndef CALCULATOR_H
#define CALCULATOR_H

extern "C" {
    int add(int a, int b);
    int subtract(int a, int b);
    int multiply(int a, int b);
    float divide(int a, int b);
}

#endif
EOF

cat > maker_so/src/calculator.cpp << 'EOF'
#include "calculator.h"

int add(int a, int b) {
    return a + b;
}

int subtract(int a, int b) {
    return a - b;
}

int multiply(int a, int b) {
    return a * b;
}

float divide(int a, int b) {
    if (b == 0) return 0;
    return (float)a / b;
}
EOF

cat > maker_so/CMakeLists.txt << 'EOF'
cmake_minimum_required(VERSION 3.10)
project(maker_so)
set(CMAKE_LIBRARY_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/lib)
include_directories(${CMAKE_SOURCE_DIR}/include)
add_library(calculator SHARED src/calculator.cpp)
EOF

# 构建项目1
cd maker_so
mkdir -p build
cd build
cmake ..
cmake --build .
cd ../..

echo -e "\n=== 项目2: 使用 libcalculator.so ==="

# 创建项目2
mkdir -p user_so/{include,src,build,lib}

# 从项目1复制 .so 和头文件到项目2
cp maker_so/lib/libcalculator.so user_so/lib/
cp maker_so/include/calculator.h user_so/include/

# 创建项目2的文件
cat > user_so/src/main.cpp << 'EOF'
#include <iostream>
#include "calculator.h"

int main() {
    int a = 15, b = 4;
    
    std::cout << "=== Using calculator library ===\n";
    std::cout << a << " + " << b << " = " << add(a, b) << std::endl;
    std::cout << a << " - " << b << " = " << subtract(a, b) << std::endl;
    std::cout << a << " * " << b << " = " << multiply(a, b) << std::endl;
    std::cout << a << " / " << b << " = " << divide(a, b) << std::endl;
    
    return 0;
}
EOF

cat > user_so/CMakeLists.txt << 'EOF'
cmake_minimum_required(VERSION 3.10)
project(user_so)
set(CMAKE_RUNTIME_OUTPUT_DIRECTORY ${CMAKE_SOURCE_DIR}/bin)
set(CMAKE_INSTALL_RPATH "${CMAKE_SOURCE_DIR}/lib")
set(CMAKE_BUILD_WITH_INSTALL_RPATH TRUE)
include_directories(${CMAKE_SOURCE_DIR}/include)
link_directories(${CMAKE_SOURCE_DIR}/lib)
add_executable(calc_app src/main.cpp)
target_link_libraries(calc_app calculator)
EOF

# 构建项目2
cd user_so
mkdir -p build
cd build
cmake ..
cmake --build .
cd ../..

echo -e "\n=== 运行结果 ==="
./user_so/bin/calc_app

echo -e "\n=== 项目文件列表 ==="
echo "项目1生成的 .so:"
ls -l maker_so/lib/
echo -e "\n项目2使用的 .so:"
ls -l user_so/lib/
echo -e "\n项目2生成的可执行文件:"
ls -l user_so/bin/

手动构建步骤(如果没有脚本)

构建项目1

bash 复制代码
cd maker_so
mkdir build && cd build
cmake ..
make
cd ../..

复制文件到项目2

bash 复制代码
mkdir -p user_so/lib user_so/include
cp maker_so/lib/libcalculator.so user_so/lib/
cp maker_so/include/calculator.h user_so/include/

构建项目2

bash 复制代码
cd user_so
mkdir build && cd build
cmake ..
make
cd ../..

运行

bash 复制代码
./user_so/bin/calc_app

验证 .so 文件

bash 复制代码
# 查看 .so 的符号
nm -D user_so/lib/libcalculator.so

# 查看可执行文件的依赖
ldd user_so/bin/calc_app

# 运行程序
./user_so/bin/calc_app

这样就完成了:

  1. 项目1 生成了 libcalculator.so
  2. 项目2 使用了这个 .so 文件
  3. 所有文件都放在各自的项目目录下,互不干扰
相关推荐
QC班长4 小时前
Maven公司私库配置踩坑点
java·服务器·maven·intellij-idea
Makoto_Kimur4 小时前
java开发面试-AI Coding速成
java·开发语言
wuqingshun3141595 小时前
说说mybatis的缓存机制
java·缓存·mybatis
知识浅谈5 小时前
DeepSeek V4 和 GPT-5.5 在同一天发布了??我也很懵,但对比完我悟了
算法
DeepModel5 小时前
通俗易懂讲透 Q-Learning:从零学会强化学习核心算法
人工智能·学习·算法·机器学习
田梓燊5 小时前
力扣:19.删除链表的倒数第 N 个结点
算法·leetcode·链表
空中海5 小时前
Kubernetes 生产实践、可观测性与扩展入门
java·贪心算法·kubernetes
猫3285 小时前
v-cloak
前端·javascript·vue.js
AC赳赳老秦6 小时前
OpenClaw二次开发实战:编写专属办公自动化技能,适配个性化需求
linux·javascript·人工智能·python·django·测试用例·openclaw
旷世奇才李先生6 小时前
Vue 3\+Vite\+Pinia实战:企业级前端项目架构设计
前端·javascript·vue.js