Linux下C++ 动态库so的生成与调用

Linux下C++ 动态库so的生成与调用

首先,动态库的生成是建立在linux下编写的函数已经可以正常调用运行,这里之进行动态库的生成以及调用需要注意的事项,并给出了示例。

单个函数

1.例如对于库函数add.cpp,extern ”C“是将其函数暴露出来

// add.cpp
extern "C" int add(int a, int b) {
    return a + b;
}

2.使用g++函数将add生成动态库

g++ -shared -o libadd.so add.cpp

3.新建main()函数,并生成的libadd.so放在main()同一级目录下

// main.cpp
#include <iostream>
#include <dlfcn.h> // 用于动态加载共享库
int main(int argc, char **argv) {


     // 打开共享库
    void* handle = dlopen("./libadd.so", RTLD_LAZY);
    
    if (!handle) {
        std::cerr << "无法加载共享库: " << dlerror() << std::endl;
        return 1;
    }
    
    // 获取共享库中的函数指针
    typedef int (*AddFunction)(int, int);
    AddFunction add = reinterpret_cast<AddFunction>(dlsym(handle, "add"));
    
    if (!add) {
        std::cerr << "无法获取函数指针: " << dlerror() << std::endl;
        return 1;
    }
    
    // 调用加法函数
    int result = add(5, 3);
    std::cout << "5 + 3 = " << result << std::endl;
    
    // 关闭共享库
    dlclose(handle);
    return 0;

}

4.编写Cmakelist

cmake_minimum_required(VERSION 2.8)#最低版本要求
project(Addition)#项目名称

set(CMAKE_CXX_STANDARD 11)#C++标准版本

add_executable(addition main.cpp)#main.cpp生成可执行addition

# 链接 libdl 库
target_link_libraries(addition dl)
target_link_libraries( addition -lpthread -lm)

5.编译并执行函数

mkdir build
cd build
cmake..
make

./addition

整体项目

对于大型项目,主函数需要的调用其他文件的函数,cmakelist中已经写好了addexcutable,并且生成的执行程序可以正常运行,现在只是简单将其附加上so文件的生成,对应操作如下。

1.首先,修改CMakeLists.txt,附加上想要生成的库,并指定链接的库

#因为需要项目包含大量的cc文件,并且在不同的文件夹中,所以使用FILE,搜索其路径下文件并保存在SRC_FILES中
FILE(GLOB SRC_FILES "src/*.cc" "src/sat/*.cc")

ADD_LIBRARY(navmg ${SRC_FILES})
#将rts.cc和其他cc文件一起生成动态库librts-lib.so;主要需要使用的函数在函数名前需要使用extern ”C“将方法暴露出来
add_library(rts-lib SHARED src/app/rts.cc ${SRC_FILES})

# 如果需要链接其他库,请添加相应的链接命令,指南写的可以将依赖的库写在同一行,但是编译通过使用错误所以分行写,知道的大佬可以解释一下

target_link_libraries(rts-lib  pthread)
target_link_libraries(rts-lib navmg )
target_link_libraries(rts-lib blas lapack)

2.动态库的调用声明及执行main函数

// main.cpp
#include <iostream>
#include <dlfcn.h> // 用于动态加载共享库

int main(int argc, char **argv) {

// 加载共享库
    void* handle2 = dlopen("/home/MZ/navmg/build/liblc-rts-lib.so", RTLD_GLOBAL |RTLD_LAZY);

    if (!handle2) {
        std::cerr << "Error loading shared library: " << dlerror() << std::endl;
        return 1;
    }


    // 获取接口函数指针
    typedef int (*RunLCRTSMain)(int, char**);
    RunLCRTSMain run_lc_rts_main = (RunLCRTSMain)dlsym(handle2, "run_lc_rts_main");
    
    if (!run_lc_rts_main) {
        std::cerr << "Error loading function: " << dlerror() << std::endl;
        dlclose(handle2);
        return 1;
    }
    
    // 调用接口函数
    int result2 = run_lc_rts_main( argc, argv);
    
    // 卸载共享库
    dlclose(handle2);
    
    return result2;
    
    return 0;

}

3.cmakelist不需要进行修改

cmake_minimum_required(VERSION 2.8)#最低版本要求
project(Addition)#项目名称

set(CMAKE_CXX_STANDARD 11)#C++标准版本

add_executable(addition main.cpp)#main.cpp生成可执行addition

# 链接 libdl 库

target_link_libraries(addition dl)
target_link_libraries( addition -lpthread -lm)

4.编译并执行函数

mkdir build
cd build
cmake..
make

./addition -p
  • 7
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux调用动态库主要是通过dlopen、dlsym、dlclose等函数来实现。下面是一个简单的示例: 1. 编写动态库代码 首先,我们需要编写一个动态库代码。这里我们以一个简单的例子来说明,假设我们需要编写一个名为libmylib.so的动态库,其中包含一个用于加法运算的函数add。 // mylib.h #ifndef MYLIB_H #define MYLIB_H #ifdef __cplusplus extern "C" { #endif int add(int a, int b); #ifdef __cplusplus } #endif #endif //MYLIB_H // mylib.cpp #include "mylib.h" int add(int a, int b) { return a + b; } 注意,在动态库中需要将函数声明为extern "C",这是因为在C++中函数名称会被编译器进行名称重整,而在动态库中需要使用原始的函数名称,所以需要使用extern "C"来告诉编译器不要进行名称重整。 2. 编译生成动态库 编译生成动态库的命令如下: g++ -shared -fPIC -o libmylib.so mylib.cpp 其中,-shared选项表示生成动态库,-fPIC选项表示编译时需要生成位置无关代码,-o选项表示指定输出文件名。 3. 编写调用动态库代码 我们可以在另一个C++程序中调用上述动态库。下面是一个简单的示例: // main.cpp #include <iostream> #include <dlfcn.h> #include "mylib.h" int main() { void *handle = dlopen("./libmylib.so", RTLD_LAZY); if (!handle) { std::cerr << "Cannot open library: " << dlerror() << '\n'; return 1; } typedef int (*add_t)(int, int); add_t add_func = reinterpret_cast<add_t>(dlsym(handle, "add")); if (!add_func) { std::cerr << "Cannot load symbol add: " << dlerror() << '\n'; dlclose(handle); return 1; } int result = add_func(2, 3); std::cout << "The result is " << result << '\n'; dlclose(handle); return 0; } 该程序首先通过dlopen函数打开动态库,并将返回的句柄保存在变量handle中。然后通过dlsym函数获取动态库中的add函数地址,并将其转换为函数指针类型add_t。最后,我们可以通过add_func指针调用add函数。 注意,在调用dlsym函数时需要指定原始的函数名称,即在编写动态库代码时使用的名称。 4. 编译生成可执行程序 编译生成可执行程序的命令如下: g++ -o main main.cpp -ldl 其中,-ldl选项表示链接动态库加载器库。 5. 运行程序 运行可执行程序的命令如下: ./main 程序的输出应该是: The result is 5 至此,我们成功地调用动态库中的函数。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值