__attribute__((constructor))用法解析

96
肉桂Kenny
2016.09.25 19:47* 字数 706

今天在看ProtocolKit的源码,看到了这么一行代码.

__attribute__((constructor)) static void _pk_extension_inject_entry(void) {

主要造成疑惑的是 __attribute__((constructor)),以前看过关于__attribute__这个关键字的,大概还记得就是可以修饰类型,函数什么的.类似一个编译标记.但是具体用法忘记了.

attribute

GNU C 的一大特色就是__attribute__ 机制。__attribute__ 可以设置函数属性(Function Attribute )、变量属性(Variable Attribute )和类型属性(Type Attribute )。

__attribute__ 书写特征是:__attribute__ 前后都有两个下划线,并切后面会紧跟一对原括弧,括弧里面是相应的__attribute__ 参数。

__attribute__ 语法格式为:__attribute__ ((attribute-list))

以上内容来自这篇文章 ,作者在文章里也说了很多详细的用法.我在这就不再重复了.下面主要说说刚刚提出的那个问题

attribute((constructor))

既然搞不懂,我的习惯是写Demo,将不懂得东西抽取出来,便于排除其他因素,
新建一个工程,将main.m改写如下:

int main(int argc, char * argv[]) {
    @autoreleasepool {
        printf("main function");
        return UIApplicationMain(argc, argv, nil, NSStringFromClass([AppDelegate class]));
    }
}
__attribute__((constructor)) static void beforeFunction()
{
    printf("beforeFunction\n");
}

然后运行,发现打印如下:

beforeFunction
main function

所以这个__attribute__((constructor))应该是在main函数之前,执行一个函数,便于我们做一些准备工作.后来,查阅了GNU的文档,印证了我的想法.

另外在文档中,还有提及,还有这么一个写法__attribute__((destructor)).文档中关于这两个用法的说明如下:

The constructor attribute causes the function to be called automatically before execution enters main (). Similarly, the destructor attribute causes the function to be called automatically after main () completes or exit () is called. Functions with these attributes are useful for initializing data that is used implicitly during the execution of the program.

大概意思是:

constructor参数让系统执行main()函数之前调用函数(被__attribute__((constructor))修饰的函数).同理, destructor让系统在main()函数退出或者调用了exit()之后,调用我们的函数.带有这些修饰属性的函数,对于我们初始化一些在程序中使用的数据非常有用.

带有优先级的参数

按照文档中所说,我们还可以给属性设置优先级.这些函数并不非要写到main.m文件中,无论写到哪里,结果都是一样的.但是,为了更显式的让阅读者看到这些定义,至少,还是在main.m文件中留个声明.

声明和实现分离的写法如下:

//声明
__attribute__((constructor(101))) void before1();

//实现
void before1()
{
    printf("before1\n");
}

下面我仅仅用作测试,就不分开写了:

static  __attribute__((constructor(101))) void before1()
{
    
    printf("before1\n");
}
static  __attribute__((constructor(102))) void before2()
{
    
    printf("before2\n");
}
static  __attribute__((constructor(102))) void before3()
{
    
    printf("before3\n");
}

上面的代码没有什么疑问.以上三个函数会依照优先级的顺序调用.另外,我以前看过,这个1-100的范围是保留的,所以,最好从100之后开始用.(但是实际上,我在项目中测试100以内的,也没有得到警告)

关于格式

按照文档中的说法,__attribute__应该放在函数声明之后,在;之前,不过这应该是个格式问题,从我之前那么多写法,也没有出错,可以得出这个结论.

每日工作总结
Web note ad 1