一、强引用
强引用:对外部目标文件的符号引用在目标文件被最终链接成可执行文件时,它们必须被正确的决议,如果没有找到该符号的定义,链接器就会报未定义错误,这种被称为强引用。典型例子如下:
/*file main.c*/#include <stdio.h> extern void test(void);int main(int argc,char* argv[]){test();printf("fun main \n");return 0;
}
编译时会报错:
$ gcc -o main main.c
/tmp/ccsWdxmr.o:在函数‘main’中:
main.c:(.text+0xb0):对‘test’未定义的引用
main.c:(.text+0xb8):对‘test’未定义的引用
collect2: 错误:ld 返回 1
二、弱引用
弱引用:对于GCC,使用__attribute__((weak))关键字来声明的函数称为弱引用。 对外部目标文件的符号引用在目标文件被最终链接成可执行文件时,如果没有找到该符号的定义,链接器不会认为它是一个错误,程序可以正常编译完成。示例如下:
/*file main.c*/#include <stdio.h> extern void __attribute__((weak)) test(void); //弱引用 test可以不实现int main(int argc,char* argv[]){if(test) test(); //对于未定义的弱引用,链接器默认值为0printf("fun main \n");return 0;
}
再次使用gcc编译,可以成功编译。只不过此时test符号为0,执行main时就不会调用函数test。编译和运行结果如下:
$ gcc -o main main.c
$ ./main
fun main
接下来我创建一个新的文件test.c,并实现test函数。
/* file test.c */#include <stdio.h>void test(){printf("fun test \n");
}
使用gcc时,通过参数”-fPIC -shared“ 来生成动态链接库libtest.so。命令如下:
$ gcc -fPIC -shared -o libtest.so test.c
再次生成可执行程序main时,通过参数“-L. -ltest“链接了libtest.so后,main文件内就有了test的占位符。那么在执行main时就会检测到函数test的存在并跳转执行,运行结果如下:
$ gcc -L. -ltest -o main main.c
$ ./main
fun test
fun main
三、弱引用的意义
弱引用的存在可以使得程序使用自定义版本的库函数;也使得程序功能更加容易剪裁和组合。
比如上面libtest.so中对test的实现是仅仅输出”fun test \n“ 。你也可以有别的类似libtest1.so 、libtest2.so等等。里面对test做你想要的实现。