linux动态库的初始化和清理

   a. Windows 中有 DllMain 入口函数, 而 Linux 中则没有。

   b. Linux 中有特殊函数 _init 和 _fini, 主要是分别用来初始化动态库和关闭的时候
      做一些必要的处理, 我们可以把自己认为需要的代码放到这两个函数里面, 它们分别
      在动态库被加载和释放的时候被执行。具体说, 如果一个动态库里面有一个名字为
      "_init" 的函数输出, 那么在第一次通过 dlopen() 函数打开这个动态库, 或者只是
      简单的作为共享动态库被打开的时候, _init 函数被自动调用执行。与之相对应的就
      是 _fini 函数, 当一个程序调用 dlclose() 去释放对这个动态库的引用的时候, 如
      果该动态库的被引用计数器为 0 了, 或者这个动态库是作为一般的共享动态库被使
      用而使用它的程序正常退出的时候, _fini就会被调用执行。

      C语言定义它们的原型如下:
      void _init(void);
      void _fini(void);

      当使用你自己的 _init 和 _fini 函数时, 会出现命名冲突, 就会得到一个
      "multiple-definition" 的错误, 编译器提示已经存在这个名字, 可以通过几种方式
      来解决:
         1). 自定义 init 函数名字, 比如 myinit 用 -Wl, 选项给 ld 传递此名字:
         gcc ... -Wl,-init=myinit

         2). 当 GCC 编译源程序时, 可以使用选项 -nostartfiles 来使共享库不与系统
             启动文件一起编译
         gcc ... -nostartfiles

         3). 使用上面的函数或 GCC 的 -nostartfiles 选项并不是很好的习惯, 因为这
             可能会产生一些意外的结果。相反, 库应该使用
             __attribute__((constructor)) 和 __attribute__((destructor)) 函数属
             性来输出它的构造函数和析构函数。如下所示:

             void __attribute__((constructor)) x_init(void);
             void __attribute__((destructor))  x_fini(void);

             构造函数会在dlopen()返回前或库被装载时调用;
             析构函数会在这样几种情况下被调用: dlclose() 返回前, 或 main() 返回
             后, 或装载库过程中 exit() 被调用时。

   c. Linux 中的初始化和释放函数不建议使用。

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Linux 系统中,动态链接库(共享库)可以定义自己的初始化清理函数,它们会在动态链接库加载和卸载时自动调用。 初始化函数的名称为 `_init`,其原型为: ```c void _init(void); ``` 清理函数的名称为 `_fini`,其原型为: ```c void _fini(void); ``` 当动态链接库加载时,动态链接器会自动调用 `_init` 函数;当动态链接库卸载时,动态链接器会自动调用 `_fini` 函数。 下面是一个简单的示例,演示如何在动态链接库中定义初始化清理函数: test.c 文件: ```c #include <stdio.h> __attribute__((constructor)) void init_func() { printf("test.so initialized\n"); } __attribute__((destructor)) void uninit_func() { printf("test.so uninitialized\n"); } void test_func() { printf("test_func called\n"); } ``` 在上述代码中,我们使用 GCC 的 `__attribute__((constructor))` 和 `__attribute__((destructor))` 属性,分别将 `init_func` 和 `uninit_func` 函数定义为初始化函数和清理函数。在 `init_func` 函数中输出初始化信息,在 `uninit_func` 函数中输出清理信息。同时,我们定义了一个 `test_func` 函数,用于在主程序中调用。 编译命令: ``` gcc -shared -fPIC -o libtest.so test.c ``` 在主程序中加载动态链接库,并调用其中的函数: ```c #include <dlfcn.h> int main() { void *handle = dlopen("./libtest.so", RTLD_NOW); if (handle == NULL) { fprintf(stderr, "dlopen failed: %s\n", dlerror()); return 1; } void (*test_func)(); test_func = (void (*)())dlsym(handle, "test_func"); if (test_func == NULL) { fprintf(stderr, "dlsym failed: %s\n", dlerror()); dlclose(handle); return 1; } test_func(); dlclose(handle); return 0; } ``` 在上述代码中,我们使用 dlopen() 函数加载动态链接库,并使用 dlsym() 函数获取其中的函数指针。然后,我们调用动态链接库中的函数,完成相应的操作。在程序结束时,动态链接器会自动调用 `_fini` 函数来清理动态链接库。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值