C 使用 dlopen 调用库中函数

C 使用 dlopen 调用库中函数,第1张

C 使用 dlopen 调用库中函数 1、打开方式:
#include 

void* native_handle = dlopen("/lib/libnative.so", RTLD_GLOBAL | RTLD_LAZY);

if (!native_handle)
{
    printf("In %s: Cannot open libnative.so, error: %sn",__FUNCTION__, dlerror());
}


// 关闭库;
int result = dlclose(native_handle);

备注:

void * dlopen( const char * pathname, int mode);

mode 是打开方式,其值有多个,不同 *** 作系统上实现的功能有所不同,在 linux 下,按功能可分为三类:

1、解析方式

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

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

2、作用范围,可与解析方式通过“|”组合使用。

RTLD_GLOBAL:动态库中定义的符号可被其后打开的其它库解析。

RTLD_LOCAL: 与 RTLD_GLOBAL 作用相反,动态库中定义的符号不能被其后打开的其它库重定位。如果没有指明是 RTLD_GLOBAL 还是 RTLD_LOCAL,则缺省为 RTLD_LOCAL。

3、作用方式

RTLD_NODELETE: 在 dlclose() 期间不卸载库,并且在以后使用 dlopen() 重新加载库时不初始化库中的静态变量。这个 flag 不是 POSIX-2001 标准。

RTLD_NOLOAD: 不加载库。可用于测试库是否已加载(dlopen()返回NULL说明未加载,否则说明已加载),也可用于改变已加载库的flag,如:先前加载库的flag为RTLD_LOCAL,用dlopen(RTLD_NOLOAD|RTLD_GLOBAL)后flag将变成RTLD_GLOBAL。这个flag不是POSIX-2001标准。

RTLD_DEEPBIND:在搜索全局符号前先搜索库内的符号,避免同名符号的冲突。这个flag不是POSIX-2001标准。

2、获取三方库中函数:
#define GET_AND_CHECK_SYMBOL(symbol_type, symbol)                             
  do {                                                                        
    if (!_##symbol_type)                                                      
      _##symbol_type =                                                        
          reinterpret_cast(dlsym(native_handle, #symbol));     
    if (!_##symbol_type) {                                                    
      printf("Cannot find symbol '%s'n", __FUNCTION__, #symbol);           
      return;                                                                 
    }                                                                         
  } while (false)

备注:

# 含义:如 #a 就是把 a 转化成字串

## 含义:如 a##b 就是把 a,b 联接起来

3、调用对方函数:
// 声明;
typedef void (*start_fp)(const char*, const char*);

// 定义变量;
start_fp _start_fp;

// 使用处;
void Start() 
{
  GET_AND_CHECK_SYMBOL(start_fp, Start);
  if (_start_fp) (*_start_fp)("temp", "temp");
}
4、注意事项 undefined symbol: xxxx.......

编译时,需加 -ldl ,否则有可能出现找不到符号的问题

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/3971498.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-10-21
下一篇 2022-10-21

发表评论

登录后才能评论

评论列表(0条)

保存