Linux如何使用SO库?

Linux系统中,SO库(Shared Object,共享库)是一种可被多个程序动态加载和调用的二进制文件,类似于Windows系统中的DLL文件,它通过代码复用减少了内存占用,便于模块化开发和程序维护,本文将详细介绍Linux下SO库的创建、使用及调试方法。

linux如何使用SO库

SO库的创建

创建SO库需先将源代码编译为位置无关代码(PIC),再链接生成共享库,假设有两个源文件add.csub.c,分别实现加法和减法函数:

// add.c
int add(int a, int b) {
    return a + b;
}
// sub.c
int sub(int a, int b) {
    return a - b;
}

编译步骤如下:

  1. 生成目标文件(添加-fPIC选项)

    gcc -fPIC -c add.c -o add.o
    gcc -fPIC -c sub.c -o sub.o

    -fPIC确保代码可被加载到任意内存地址,是共享库的必要选项。

  2. 链接生成SO库

    gcc -shared -o libmymath.so add.o sub.o

    -shared指定生成共享库,libmymath.so是库文件名(Linux下共享库通常以lib开头,.so

SO库的使用方式

SO库的使用分为静态链接(编译时链接)和动态链接(运行时加载)两种。

静态链接(编译时链接)

在编译可执行文件时,通过-L(指定库路径)和-l(指定库名)选项链接SO库,使用libmymath.somain.c

linux如何使用SO库

// main.c
#include <stdio.h>
extern int add(int, int);
extern int sub(int, int);
int main() {
    printf("3+5=%dn", add(3, 5));
    printf("3-5=%dn", sub(3, 5));
    return 0;
}

编译命令:

gcc main.c -L. -lmymath -o main
  • -L.:表示当前目录存在SO库(若库在/usr/lib等系统路径,可省略-L)。
  • -lmymath:链接libmymath.so(去掉lib前缀和.so后缀)。

运行时,需确保SO库在系统默认路径(如/usr/lib)或通过LD_LIBRARY_PATH指定路径:

export LD_LIBRARY_PATH=.:$LD_LIBRARY_PATH  # 添加当前路径到动态库搜索路径
./main

动态链接(运行时加载)

通过dlopendlsymdlclose等函数在程序运行时动态加载SO库,适用于插件化开发,修改main.c如下:

// main.c (动态加载版)
#include <stdio.h>
#include <dlfcn.h>
int main() {
    void *handle = dlopen("./libmymath.so", RTLD_LAZY);
    if (!handle) {
        fprintf(stderr, "dlopen error: %sn", dlerror());
        return 1;
    }
    int (*add_ptr)(int, int) = dlsym(handle, "add");
    int (*sub_ptr)(int, int) = dlsym(handle, "sub");
    if (!add_ptr || !sub_ptr) {
        fprintf(stderr, "dlsym error: %sn", dlerror());
        dlclose(handle);
        return 1;
    }
    printf("3+5=%dn", add_ptr(3, 5));
    printf("3-5=%dn", sub_ptr(3, 5));
    dlclose(handle);
    return 0;
}

编译时需链接dl库:

gcc main.c -ldl -o main_dynamic

运行时无需预先设置LD_LIBRARY_PATH(除非SO库不在默认路径),直接执行./main_dynamic即可。

SO库的查看与调试

查看依赖库

使用ldd命令检查可执行文件依赖的共享库:

ldd ./main

输出示例:

        linux-vdso.so.1 =>  (0x00007ffc...)
        libmymath.so => ./libmymath.so (0x00007f8c...)
        libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f8c...)
        /lib64/ld-linux-x86-64.so.2 (0x00007f8c...)

查看库符号

使用nm命令查看SO库中的导出符号(函数/变量):

linux如何使用SO库

nm -D libmymath.so

输出示例:

0000000000001120 T add
0000000000001130 T sub
U __libc_start_main
U printf

T表示全局符号,U表示未定义符号(需依赖其他库)。

更新共享库缓存

将SO库复制到系统库路径(如/usr/lib)后,需运行ldconfig更新缓存:

sudo cp libmymath.so /usr/lib/
sudo ldconfig

之后无需设置LD_LIBRARY_PATH即可直接运行程序。

静态链接与动态链接对比

特性 静态链接 动态链接
加载时机 编译时 运行时
文件大小 可执行文件较大(包含库代码) 可执行文件较小(仅包含链接信息)
更新维护 需重新编译程序 仅需替换SO库,程序无需重新编译
依赖管理 无外部依赖 需确保SO库存在于系统路径
内存占用 多个程序无法共享库代码 多个程序共享同一份库代码,节省内存

相关问答FAQs

Q1: 运行时报错“cannot open shared object file: No such file or directory”怎么办?
A: 此错误表示程序找不到SO库,解决方法:

  1. 检查SO文件是否存在(如libmymath.so);
  2. 使用export LD_LIBRARY_PATH=库路径:$LD_LIBRARY_PATH临时添加库路径;
  3. 将SO库复制到系统默认路径(如/usr/lib)并运行sudo ldconfig更新缓存;
  4. 检查ldd输出中库路径是否正确。

Q2: 编译时报错“undefined reference to”符号错误,如何解决?
A: 此错误通常是因为未正确链接SO库或符号未导出,解决方法:

  1. 确认编译时添加-l库名选项(如-lmymath);
  2. 检查SO库是否使用-shared生成,且目标文件编译时添加了-fPIC
  3. 使用nm -D lib库名.so检查符号是否存在,若符号未导出(显示为U),需在源代码中使用__attribute__((visibility("default")))修饰函数,或检查链接脚本设置。

原创文章,发布者:酷番叔,转转请注明出处:https://cloud.kd.cn/ask/23924.html

(0)
酷番叔酷番叔
上一篇 2025年9月16日 11:35
下一篇 2025年9月16日 11:52

相关推荐

  • Linux中如何查看显卡型号?常用命令与操作方法有哪些?

    在Linux系统中,查看显卡型号是系统管理、驱动安装或性能优化时的常见需求,由于Linux显卡驱动分为开源驱动(如Mesa、nouveau)和闭源驱动(如NVIDIA-Linux-x86、amdgpu-pro),不同场景下查看方法可能略有差异,本文将详细介绍通过命令行工具、系统文件及图形界面查看显卡型号的多种方……

    2025年10月3日
    1100
  • 你知道吗?90%的人不知道的真相?

    前提条件安装GCC编译器Linux默认不包含C编译器,需安装GCC(GNU Compiler Collection):sudo apt update && sudo apt install gcc # Debian/Ubuntusudo yum install gcc # CentOS/RHEL……

    2025年7月20日
    5200
  • Linux下如何编写DLL动态链接库文件?

    在Linux操作系统中,并没有Windows环境下的动态链接库(DLL)文件格式,但Linux提供了功能类似的共享库(Shared Object,文件后缀为.so),它允许程序在运行时动态加载和链接,实现代码复用和模块化开发,编写Linux下的共享库(即“Linux DLL”)需要遵循特定的编译和链接流程,本文……

    2025年10月1日
    1800
  • 如何执行Linux U盘启动?详细步骤方法是什么?

    执行Linux U盘启动是许多用户在安装Linux系统、进行数据恢复或体验Linux发行版时的常用操作,其核心原理是将Linux系统镜像写入U盘,使U盘具备引导计算机的能力,整个过程可分为准备工作、制作启动U盘、BIOS/UEFI设置、启动Linux及后续处理几个关键环节,每个环节都需要细致操作以确保成功,准备……

    2025年10月1日
    1700
  • Linux下如何通过TCP搜索指定端口进程?

    在Linux系统中,排查端口占用问题或定位特定端口对应的进程是日常运维和开发中的常见需求,当需要查找某个TCP端口被哪个进程占用时,可以通过多种命令实现,每种命令的适用场景和输出格式略有不同,本文将详细介绍这些方法及其使用技巧,使用netstat命令查找端口进程netstat是传统的网络状态查看工具,虽然在新版……

    2025年10月5日
    1200

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-880-8834

在线咨询: QQ交谈

邮件:HI@E.KD.CN

关注微信