我是靠谱客的博主 贪玩飞鸟,最近开发中收集的这篇文章主要介绍__attribute__((weak)),觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

情况是这样的,碰到一个棘手的问题:我们不确定外部模块是否提供一个函数func,但是我们不得不用这个函数,即自己模块的代码必须用到func函数:

extern int func(void);

...................

int a = func();

if( a > .....)

{

..........

}

............

我们不知道func函数是否被定义了

这会导致2个结果:

1:外部存在这个函数func,并且EXPORT_SYMBOL(func),那么在我自己的模块使用这个函数func,正确。

2:外部其实不存在这个函数,那么我们使用func,程序直接崩溃。

 

所以这个时候,__attribute__((weak)) 派上了用场。

在自己的模块中定义:

int  __attribute__((weak))  func(......)

{

return 0;

}

 

将本模块的func转成弱符号类型,如果遇到强符号类型(即外部模块定义了func),那么我们在本模块执行的func将会是外部模块定义的func。

如果外部模块没有定义,那么,将会调用这个弱符号,也就是在本地定义的func,直接返回了一个1(返回值视具体情况而定)

相当于增加了一个默认函数。

原理:连接器发现同时存在弱符号和强符号,有限选择强符号,如果发现不存在强符号,只存在弱符号,则选择弱符号。如果都不存在:静态链接,恭喜,编译时报错,动态链接:对不起,系统无法启动。 

 

TIPS

weak属性只会在静态库(.o .a )中生效,动态库(.so .ko(内核模块))中不会生效。

 

举个例子:

strong.c  //生成libstrong.so

 

#include <stdio.h>

void real_func()
{
	printf("int real funcn");
}


weak.c //生成libweak.so

 

 

#include <stdio.h>
void real_func() __attribute__((weak));


void real_func()
{
	printf("fake funcn");
}


main.c //

 

 

#include <stdio.h>
extern void real_func();

void main()
{
        real_func();
}

 

 

 

 

 

如果 

gcc main.c -lstrong -lweak

那么输出结果"real func"。

 

如果 

gcc main.c -lweak -lstrong

那么输出结果为"fake func"。

 

可见,对于动态库,weak属性毫无作用,且main中调用哪个real_func(),取决于链接的顺序。

 

如果将strong.c 和 weak.c编译成.a或者.o

gcc main.c strong.o weak.o

或者

gcc main.c strong.o weak.o

那么输出结果都是"real func"。

所以,如果在so中使用weak属性,那么任何不符合预期的情况,都是可能出现的。

官方解释:

https://sourceware.org/bugzilla/show_bug.cgi?id=3946

如果你觉得有用,请打赏http://39.98.242.44/

最后

以上就是贪玩飞鸟为你收集整理的__attribute__((weak))的全部内容,希望文章能够帮你解决__attribute__((weak))所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(41)

评论列表共有 0 条评论

立即
投稿
返回
顶部