linux环境下so编程

本文通过一个简单的例子,介绍linux环境下动态库编程的基本做法。

基本概念

库从本质上来说是一种可执行代码的二进制格式,可以被载入到内存中执行。库分为静态库和动态库两种。

在Linux下,静态库的名字一般为libxxx.a,其中xxx为库的名字,使用静态库时,整个库的所有数据都会被整合到目标代码中,因而文件比较大,执行时不再需要外部库的支持,如果库函数有变化,程序必须重新编译。

动态库的名字一般是libxxx.M.N.so,其中xxx为库名,M是主版本号,N是副版本号,版本号是可选的,但名字必须要有。动态库在编译时并没有被编进目标代码中,程序执行到相关函数时才调用库中的相应函数,因此可执行文件较小,程序运行环境必须提供相应的动态库。动态库的修改不影响程序,因而升级比较方便。

对于静态库,链接器会找出程序需要的函数,将它们拷到执行文件,由于这种拷贝是完整的,一旦链接成功,静态库也就不再需要了。对于动态库,则会在程序内打个标记指明程序运行时,要先载入这个库。由于动态库节省空间,在链接时会优先去链动态库,即如果同时存在静态库和动态库,不特别指定的话,将与动态库链接。

制作动态库

为方便演示,这里做个简单的加减法库,代码包括add.h,add.c,sub.h,sub.c共4个文件。

// === add.h ===
#ifndef _ADD_H_
#define _ADD_H_

int add(int, int);

#endif

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

// === sub.h ===
#ifndef _SUB_H_
#define _SUB_H_

int sub(int, int);

#endif

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

通过以下命令生成动态库文件。

gcc -o libtest.so add.c sub.c -fPIC -shared -g -Wall -O0

可以用nm -D libtest.so查看导出了哪些接口,可以看到,add和sub接口已正常导出,可以使用。

$ nm -D libtest.so
                 w _ITM_deregisterTMCloneTable
                 w _ITM_registerTMCloneTable
0000000000201020 B __bss_start
                 w __cxa_finalize
                 w __gmon_start__
0000000000201020 D _edata
0000000000201028 B _end
00000000000005c0 T _fini
0000000000000480 T _init
000000000000059a T add
00000000000005ae T sub

静态链接动态库

动态库有了,下面写个程序来调用动态库,假设文件为call.c,代码如下。

#include <stdio.h>
#include "add.h"
#include "sub.h"
int main()
{
    int x, y;
    while (~scanf("%d%d", &x, &y))
    {
        printf("add=%d, sub=%d\n", add(x,y), sub(x,y));
    }
    return 0;
}

运行以下命令进行编译链接,其中-L参数指定需要链接的so库的位置,-l参数接库名指定链接哪个库。

gcc -o call call.c -L. -ltest

当然,默认情况下call程序还不能运行,因为它找不到libtest.so库,可以用ldd call看下它静态链接了哪些库。

$ ldd call
    linux-vdso.so.1 (0x00007fffe7acc000)
    libtest.so => not found
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fd695350000)
    /lib64/ld-linux-x86-64.so.2 (0x00007fd695943000)

可以看到,libtest.so库确实没找到,因为编译时能找到so并不代表运行时就能找到,编译时如果so文件不在系统默认的lib目录下,则需要用-L参数指定位置;而运行时加载so可以通过多种方式指定,按先后顺序依次如下:

  • 环境变量LD_LIBRARY_PATH
  • 链接时-rpath指定的共享库查找路径
  • ldconfig配置指定的路径
  • /lib, /usr/lib目录

关于ldconfig多说一句,在Linux系统中,动态链接库的配置文件一般在/etc/ld.so.conf文件内,该文件会包含/etc/ld.so.conf.d目录。当修改了ld.so.conf或者/ld.so.conf.d目录下的文件,或者往该目录下拷贝了新的动态库时,要执行ldconfig命令,它负责搜索/lib, /usr/lib以及/etc/ld.so.conf中所列的目录下可用的动态链接库文件到缓存/etc/ld.so.cache中。执行ldconfig时,如果带了目录,则表示在缓存/etc/ld.so.cache中追加指定目录下的共享库;如果是单独运行,则它只会搜索/lib, /usr/lib和/etc/ld.so.conf列出的目录,重建/etc/ld.so.cache。

在本例中,可以通过多种方式让程序能够跑起来。

  • 改环境变量:export LD_LIBRARY_PATH=.
  • 运行ldconfig $(pwd),需root权限
  • 改编译指令,追加参数-Wl,-rpath=.
  • 将libtest.so拷贝到/lib或/usr/lib目录下。

动态链接动态库

Linux提供了一组api用于动态加载so文件。

#include <dlfcn.h>
void* dlopen(const char *filename, int flag);
char* dlerror(void);
void* dlsym(void *handle, const char *symbol);
int dlclose(void *handle);

说明:

  • filename如果不以/开头,则为相对路径,将按照以下顺序查找。
    (1)环境变量LD_LIBRARY
    (2)/etc/ld.so.cache
    (3)/lib, /usr/lib
  • flag表示在什么时候解决未定义的符号调用,可取RTLD_LAZY和RTLD_NOW。
    下面是个动态链接的例子。
#include <stdio.h>
#include <dlfcn.h>
typedef int (*Func)(int, int);
int main()
{
    void *handle = dlopen("libtest.so", RTLD_LAZY);
    if (dlerror()) perror("dlopen");
    Func fnadd = dlsym(handle, "add");
    if (dlerror()) perror("load add");
    Func fnsub = dlsym(handle, "sub");
    if (dlerror()) perror("load sub");

    int x, y;
    while (~scanf("%d%d", &x, &y))
        printf("%d %d\n", fnadd(x, y), fnsub(x, y));
    
    dlclose(handle);
    return 0;
}

注意,如果采用动态加载方式,编译时须指定-rdynamic -ldl选项。

gcc -o call call.c -rdynamic -ldl
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,458评论 6 513
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,030评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,879评论 0 358
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,278评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,296评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,019评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,633评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,541评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,068评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,181评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,318评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,991评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,670评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,183评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,302评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,655评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,327评论 2 358

推荐阅读更多精彩内容