浅谈C语言中的强符号、弱符号、强引用和弱引用【转】
转自:http://www.jb51.net/article/56924.htm
首先我表示很悲剧,在看《程序员的自我修养--链接、装载与库》之前我竟不知道C有强符号、弱符号、强引用和弱引用。在看到3.5.5节弱符号和强符号时,我感觉有些困惑,所以写下此篇,希望能和同样感觉的朋友交流也希望高人指点。
首先我们看一下书中关于它们的定义。
引入场景:(1)文件A中定义并初始化变量i(int i = 1), 文件B中定义并初始化变量i(int i = 2)。编译链接A、B时会报错b.o:(.data+0x0): multiple definition of `i';a.o:(.data+0x0): multiple definition of `i'。(2)在文件C中定义并初始化两个变量i(int i = 1; int i = 2), 编译链接时会报错c.c:2:5: error: redefinition of ‘i'; c.c:1:5: note: previous definition of ‘i' was here。
强符号:像场景中这样的符号定义被称为强符号,对于C/C++来说,编译器默认函数和初始化的全局变量为强符号。
弱符号:接上文,为初始化的全局变量为弱符号。
编译器关于强弱符号的规则有:(1)强符号不允许多次定义,但强弱可以共存;(2)强弱共存时,强覆盖弱;(3)都是弱符号时,选择占用空间最大的,如选择 double类型的而不选择int类型的。
由以上定义所以有我之前没有想到的场景:
代码a.c:
1 int i = 2;
代码b.c:
#include<stdio.h>
int i;
int main(int argc, char** argv)
{
printf("i = %d\n", i);
return 0;
}
编译文件a和b并链接,结果输出i为2而不是0。
并且在同一个文件中定义但未初始化两个相同的变量不会报错,只有在使用变量时才会报错。
对于GCC编译器来说,还允许使用__attribute__((weak))来将强符号定义为弱符号,所已有
代码c.c
#include<stdio.h>
__attribute__((weak)) int i = 1;
int main(int argc, char** argv)
{
printf("i = %d\n", i);
return 0;
}
结果i的输出仍未2而不是1。
那么对于函数而言是不是也这样呢?先不看函数,而是先看由强弱符号而进一步引入的强弱引用。书中关于强弱引用的概述是对于强引用若未定义则链接时肯定会报错,而对于弱引用则不会报错,链接器默认其为0(这一点对于函数好理解,即函数符号所代表入口地址为0;对于变量就要注意了,既然是引用那自然就是地址了,所以同函数一样变量的地址为0而不是变量的值为0)。此时对于强弱引用是不是还没有什么明确的概念呢?到底什么是引用?引用和符号又是什么关系?这里我说一下我的理解(欢迎指正),在定义和声明处指定的函数名、变量名即为对应的符号,而在代码其他处调用函数或使用变量时,则把函说明和变量名看作引用,这样一来符号和引用在代码层面上其实就是一个东西,只是根据环境而叫法不同而已。那么强符号对应强引用,弱符号对应弱引用。
有上面的强弱引用的特点可看出,当一个函数为弱引用时,不管这个函数有没有定义,链接时都不会报错,而且我们可以根据判断函数名是否为0来决定是否执行这个函数。这样一来,包含这些函数的库就可以以模块、插件的形式和我们的引用组合一起,方便使用和卸载,并且由于强符号可以覆盖弱符号和强弱符号与强弱引用的关系可知,我们自己定义函数可以覆盖库中的函数,多么美妙。
先看根据条件判断是否执行函数:
代码d.c
#include<stdio.h>
void func()
{
printf("func()#1\n");
}
代码e.c
#include<stdio.h>
__attribute__((weak)) void func();
int main(int argc, char** argv)
{
if (func)
func();
return 0;
}
编译d.c,cc -c d.c 输出d.o;编译e.c并链接d.o,cc d.o e.c -o e输出可执行文件e,运行e正常执行函数func。编译e.c但不链接d.o,此时并不会报错,只不过func不会执行,因为没有它的定义所以if(func)为假。
再看函数覆盖:
代码f.c
#include<stdio.h>
__attribute__((weak)) void func()
{
printf("func()#1\n");
}
代码g.c
#include<stdio.h>
void func()
{
printf("func()#2\n");
}
int main(int argc, char** argv)
{
func();
return 0;
}
~
编译链接,结构输出"func()#2"。
以上可以说明函数和变量是保持一致的,其实对应变量也可以像使用函数那样先判断再使用,只不过不是判断变量的值而是变量的地址,如
代码v1.c
int i = 2;
代码v2.c
#include<stdio.h>
__attribute__((weak)) extern int i;
int main(int argc, char** argv)
{
if (&i)
printf("i = %d\n", i);
return 0;
}
~
编译并链接v1时,输出2;编译但不链接v1时无输出。这样做时要分清定义和声明的区别,__attribute__((weak)) int i 是定义变量并转换为弱符号,这样i是分配了空间的,而__attribute__((weak)) extern int i 则将原来定义的变量i由强符号转换为弱符号,导致使用i时不是强引用而是弱引用。不过虽然变量可以这么做但没有函数那样有意义。
上面关于强弱引用仍旧使用的是GCC提供的__attribute__((weak)),而书中还提到了__attribute__((weakref)),后者貌似更能体现“引用”这一关键词。而我之所以使用前者来介绍强弱引用,是因为我对关于强弱符号与强弱引用对应关系的理解。关于__attribute__((weakref))的使用方法,这里介绍一种(两者都有不同的使用方法)。
代码a.c
#include<stdio.h>
void bar()
{
printf("foo()\n");
}
代码b.c
#include<stdio.h>
static void foo() __attribute__((weakref("bar")));
int main(int argc, char** argv)
{
if (foo)
foo();
return 0;
}
注意函数foo的static修饰符,没有的话会报错,这样将函数foo限制在只有本文件内可使用。
浅谈C语言中的强符号、弱符号、强引用和弱引用【转】相关推荐
- c语言弱符号与函数指针,浅谈C语言中的强符号、弱符号、强引用和弱引用【转】...
首先我表示很悲剧,在看<程序员的自我修养--链接.装载与库>之前我竟不知道C有强符号.弱符号.强引用和弱引用.在看到3.5.5节弱符号和强符号时,我感觉有些困惑,所以写下此篇,希望能和同样 ...
- c语言指针很危险,浅谈C语言中指针使用不当的危险性.doc
浅谈C语言中指针使用不当的危险性.doc 第 19 卷 Vol . 19 第 2 期 No . 2 洛阳师专学报 Journal of Luoyang Teachers College 2000 年 ...
- c程序语言的常量变量和标识符,浅谈C语言中的常量与变量.pdf
课程教育研究 CourseEducationResearch 2014年4月 上旬刊 教学.信息 浅谈C语言中的常量与变量 刘 星 (青 岛工学院商学院 山东 青岛 266300) [摘要]在任何一种 ...
- c语言中很多中括号由外向里,浅谈C语言中的类型声明
文章目录 [隐藏] 新年第一更!之前群友问了一个 C语言 问题,即int(*(*p)()).int *(*p)()和int *(*p())的区别在哪里.确实,有时C语言的类型声明是很魔性的,看着也很令 ...
- c语言如何求一个数学表达式的值,浅谈C语言中表达式的求值
该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 C语言研究性学习的路线 现行的多数C语言教材有太多的误区,不仅不能给读者提供有效的学习线索,还常常"误导"读者,于是,"死记 ...
- 计算机语言中的次方,浅谈Go语言中的次方用法
Go语言中符号 " ^ " 不再用于次方,而是表示"按位异或的运算" 具体的运算规则如下: 按位异或 ^ : 两位一个为 0, 一个为 1 ,结果为 1 ,否则 ...
- 如何求c语言表达式的值,浅谈C语言中表达式的求值
该楼层疑似违规已被系统折叠 隐藏此楼查看此楼 C语言研究性学习的路线 现行的多数C语言教材有太多的误区,不仅不能给读者提供有效的学习线索,还常常"误导"读者,于是,"死记 ...
- 浅谈c语言中的字符串
写在前面:最近MM问了我一个问题--字符串在内存中位于哪里?我想当然是位于数据段(data segment).她又问,那怎么保证它只读呢?我答不上了.有些问题,看似简单,背后却隐藏着天机,后来查了一些 ...
- linux ruby作用域,浅谈ruby语言中的一些概念(lambda, proc, block)
block 和 Proc ruby中的block是方法的一个重要但非必要的组成部分,任何方法后面都可以挂载一个block,如果你定义的方法想使用block做点事情.那么你需要使用yield关键字或者& ...
- 浅谈C语言中数组理解
前言:本人为c语言初学者,如果引用理解错误请各位指出!我定会虚心改正,和大家共同前行. 目录 1.初始化数组 2.给数组元素赋值 3.数组边界 4.指定数组的大小 5.多维数组 6.关于二维数组的初始 ...
最新文章
- 软件工程师学硬件-怎么看原理图之协议类接口UART
- 小学期学习记录(一)
- 勒索软件指向Flash与Silverlight漏洞
- Go语言学习之GOPATH
- VMware workstation和ESXi的区别
- iOS-属性字符串添加下划线、删除线
- 根据指定字段排序编号(SQL Server 2005,Update,Order By)
- 【BZOJ】1176: [Balkan2007]Mokia(cdq分治)
- 自动驾驶1-7: 进入自动驾驶汽车行业的建议Advice for Breaking into the Self-Driving Cars Industry
- houdini pyro
- python扩展模块开发
- 计算机win10无法打开小键盘,win10数字小键盘开机不启动解决方法
- 环信即时通信聊天理解
- u盘插在linux显示不了,U盘在Linux下显示不正常的解决方法
- 云原生GIS技术全解读
- 华为服务器TS200-2280安装HBA卡
- Flink调优(一)资源调优、背压问题的分析
- vue移动端长按文字、图片阻止浏览器默认行为
- 汉字Unicode编码
- 使用IE浏览器下载文件,文件名乱码问题
热门文章
- 动态链接库(共享库).so文件的使用
- php 导出文件另行指定路径,生成excel文件到指定目录的函数php类库
- 例2.11 FatMouse' Trade - 九度教程第21题(贪心算法)
- conda install 等价与pip install -e .
- 中南大学计算机学院伍逸凡,关于公布2017年湖南省大学生力学竞赛等15项学科竞赛结果的通知...
- 版本更新带来的缓存问题_【第1563期】缓存最佳实践 amp; maxage的陷阱
- 用Java发起HTTP请求与获取状态码(含状态码列表)
- 一、K3 Wise 实施指导《K3 Wise实施手册》
- 学会写出"图形界面+数据库"的程序要多长时间?
- MyEclipse中用Maven创建Web项目(亲测有效)