天天看点

linux共享库的动态加载

linux c中打开库文件,用的是dlopen函数,该函数主要是加载库中的符号,

第一个参数:是共享库的名称,注意如果要引用本文件夹内的库文件,加./文件名

第二个参数:

RTLD_NOW:在dlopen返回前,对于动态库中的未定义的符号不执行解析(只对函数引用有效,对于变量引用总是立即解析)。

RTLD_NOW: 需要在dlopen返回前,解析出所有未定义符号,如果解析不出来,在dlopen会返回NULL,错误为:: undefined symbol: xxxx.......

void*dlsym(void*handle,constchar*symbol)

根据动态链接库操作句柄与符号,返回符号对应的地址

handle:由dlopen打开动态链接库后返回的指针;

symbol:要求获取的函数或全局变量的名称。

void* 指向函数的地址,供调用使用。

用法:

1.创建一个目录, mycal

mkdir mycal

2.在mycal目录下创建两个目录 modules, src

3.在modules创建两个文件 cal.h,cal.c

cal.h:

#ifndef __CAL_H__

#define __CAL_H__

int add(int a, int b);

int sub(int a, int b);

int dive(int a, int b);

int mul(int a, int b);

#endif

cal.c:

#include "cal.h"

int add(int a, int b)

{

    return a + b;

}

int sub(int a, int b)

{

    return a - b;

}

int mul(int a, int b)

{

    a * b;

}

int dive(int a, int b)

{

    return a / b;

}

制作共享库:

gcc -c -fPIC cal.c

gcc -shared -Wl,-soname,libmycal.so.1 -o libmycal.so.1.10 cal.o

soname是一个符号链接的名字,只包含共享库的主版本号,想要深入理解相关细节,请查阅相关资料,这里就不多说了

注意:-Wl,-soname,libmycal.so.1 之间没有空格

4.在src目录下创建 main.c文件

main.c:

#include <stdio.h>

#include "../modules/cal.h"

#include <dlfcn.h>

int main(void)

{

    int (*add)(int a, int b);

    void *handle = dlopen("../modules/libmycal.so.1.10", RTLD_LAZY);

    if (handle == NULL) {

        printf("Failed load library\n"); 

        return -1;

    }

    char *err = dlerror();

    if (err != NULL) {

        printf("%s\n", err); 

        return -1;

    }

    add = dlsym(handle, "add");

    err = dlerror();

    if (err != NULL) {

        printf("%s\n", err); 

        dlclose(handle);

        return -1;

    }

    printf("%d", add(5, 3));

    dlclose(handle);

    return 0;

}

编译文件:

gcc -o main main.c -ldl

注意:-ldl选项在最后

至此共享库的动态加载算是完成了,谢谢大家!!!

继续阅读