risc-v中文社区

 找回密码
 立即注册
查看: 1192|回复: 1

linux动态库的初始化和清理(init()函数和fini()函数)

[复制链接]

347

主题

564

帖子

2237

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
2237
发表于 2022-9-4 20:00:47 | 显示全部楼层 |阅读模式
                    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 中的初始化和释放函数不建议使用。
               

回复

使用道具 举报

347

主题

564

帖子

2237

积分

管理员

Rank: 9Rank: 9Rank: 9

积分
2237
 楼主| 发表于 2022-9-4 20:03:41 | 显示全部楼层
#include <stdio.h>
#include <stdlib.h>
#include <sys/time.h>
#include <signal.h>
#include <unistd.h>

int main(void) {
    printf("!!!Hello World!!!\n"); /* prints !!!Hello World!!! */
    while(1)
        pause();
    return EXIT_SUCCESS;
}
__attribute__((constructor(30))) void beforefunction()
{
    printf("fun run befor main\n");
}
__attribute__((constructor(11))) void beforefunction2() //数值越小越先执行
{
    printf("fun2 run befor main\n");
}
执行显示:
fun2 run befor main
fun run befor main
!!!Hello World!!!


回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



Archiver|手机版|小黑屋|risc-v中文社区

GMT+8, 2024-4-19 17:51 , Processed in 0.017692 second(s), 17 queries .

risc-v中文社区论坛 官方网站

Copyright © 2018-2021, risc-v open source

快速回复 返回顶部 返回列表