Linux下C调用静态库和动态库

本文主要介绍 Linux 下 C 语言调用静态库和动态库。

1. 概念

动态库和静态库二者的不同点在于代码被载入的时刻不同。

静态库的代码在编译过程中已经被载入可执行程序,因此体积比较大。动态库(共享库)的代码在可执行程序运行时才载入内存,在编译过程中仅简单的引用,因此代码体积比较小。

静态情况下,把库直接加载到程序中。而动态库链接的时候,它只是保留接口,将动态库与程序代码独立,这样就可以提高代码的可复用度,和降低程序的耦合度。

静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需要动态库存在。

2. 样例文件

welcome.c:

#include <stdio.h>
#include "welcome.h"

void welcome()
{
    printf("welcome to my code world!\n");
}

这是一个样例程序,打印一句话。

welcome.h:

#ifndef _WELCOME_H
#define _WELCOME_H

void welcome();

#endif 

为上一个文件的声明。

3. 静态库

生成静态库文件:

$ gcc -Wall -O2 -fPIC -I./ -c -o welcome.o welcome.c
$ ar crv libwelcome.a welcome.o

ar 命令的参数如下:

参数意义
-r将 objfile 文件插入静态库尾或者替换静态库中同名文件
-x从静态库文件中抽取文件 objfile
-t打印静态库的成员文件列表
-d从静态库中删除文件 objfile
-s重置静态库文件索引
-v创建文件冗余信息
-c创建静态库文件

test-sta.c 代码如下:

#include <stdio.h>

int main(void)
{
    welcome();

    return 0;
}

编译:

$ gcc test-sta.c -o test-sta ./libwelcome.a
或者
$ gcc test-sta.c -o test-sta -L. -lwelcome 
再或者
$ gcc test-sta.c -o test-sta -L. libwelcome.a

注:
这里的 -l/ 文件名,-L/ 路径,是通过 -I 和 -L 指定对应的库文件名和库文件的路径,这里就是当前目录,libwelcome.a 就是要用的静态库,这样对应的静态库已经编译到对应的可执行程序中。执行对应的可执行文件便可以得到对应函数调用的结果。在 main.c 中不需要包含导出文件的头文件。

运行:

$ ./test-sta 
welcome to my code world!
4. 动态库

生成动态库文件:

$ gcc -o2 -fPIC -shared welcome.c -o libwelcome.so

或者

$ gcc -o2 -fPIC -c welcome.c 
$ gcc -shared -o libwelcome.so welcome.o 

其中:

  • fPIC:产生与位置无关代码,全部使用相对地址。
  • shared:生成动态库。
(1). 编译时加载(隐式)

test-implicit.c 代码如下:

#include <stdio.h>

int main()
{
    welcome();

    return 0;
}

和静态库一样,测试代码不需要包含导出函数的头文件。

编译:

$ gcc -o2 -Wall -L. -lwelcome test-implicit.c -o test-implicit

查看 test-implicit 动态段信息,发现已经依赖 libwelcome.so:

$ ldd test-implicit 
    linux-vdso.so.1 =>  (0x00007f0902951000)
    libwelcome.so => ./libwelcome.so (0x00007f090274f000)
    libstdc++.so.6 => /usr/lib64/libstdc++.so.6 (0x0000003548600000)
    libm.so.6 => /lib64/libm.so.6 (0x000000353de00000)
    libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x0000003548200000)
    libc.so.6 => /lib64/libc.so.6 (0x000000353da00000)
    /lib64/ld-linux-x86-64.so.2 (0x000000353d600000)

若此时直接运行,会提示找不到动态库:

$ ./test-implicit
$ ./test-implicit: error while loading shared libraries: libwelcome.so: cannot open shared object file: No such file or directory

可以通过下列三种方法解决:

  • 方法一:修改环境变量

    $ export LD_LIBRARY_PATH=$(pwd):$LD_LIBRARY_PATH
    
  • 方法二:将库文件链接到系统目录下

    $ ln -s ./libwelcome.so /usr/lib
    
  • 方法三:修改 /etc/ld.so.conf

    $ sudo echo $(pwd) >> /etc/ld.so.conf
    $ sudo ldconfig
    

再次运行:

$ ./test-implicit
welcome to my code world!
(2). 运行时链接(显式)

test-explicit.c 代码如下:

#include <stdio.h>
#include <dlfcn.h>

#define LIB "./libwelcome.so"

int main(void)
{
    /*
     * RTLD_NOW:将共享库中的所有函数加载到内存 
     * RTLD_LAZY:会推后共享库中的函数的加载操作,直到调用dlsym()时方加载某函数
     */

    void *dl = dlopen(LIB,RTLD_LAZY); //打开动态库

    if (dl == NULL)
        fprintf(stderr,"Error:failed to load libary.\n");

    char *error = dlerror(); //检测错误
    if (error != NULL)
    {
        fprintf(stderr,"%s\n",error);
        return -1;
    }

    void (*func)() = dlsym(dl,"welcome"); // 获取函数地址
    error = dlerror(); //检测错误
    if (error != NULL)
    {
        fprintf(stderr,"%s\n",error);
        return -1;
    }

    func(); //调用动态库中的函数

    dlclose(dl); //关闭动态库
    error = dlerror(); //检测错误
    if (error != NULL)
    {
        fprintf(stderr,"%s\n",error);
        return -1;
    }

    return 0;
}

编译:

$ gcc -ldl test-explicit.c -o test-explicit

查看 test-explicit 动态段信息,没有发现依赖 libwelcome.so:

$ ldd test-explicit
    linux-vdso.so.1 =>  (0x00007ffed89e5000)
    libdl.so.2 => /lib64/libdl.so.2 (0x000000353e600000)
    libstdc++.so.6 => /usr/lib64/libstdc++.so.6 (0x0000003548600000)
    libm.so.6 => /lib64/libm.so.6 (0x000000353de00000)
    libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x0000003548200000)
    libc.so.6 => /lib64/libc.so.6 (0x000000353da00000)
    /lib64/ld-linux-x86-64.so.2 (0x000000353d600000)

运行:

$ ./test-explicit
welcome to my code world!

区别:隐式调用在编译可执行程序时需要指定库文件的搜索路径,而显式调用编译可执行程序时不用加上。

  • 3
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值