- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
似乎未初始化的全局变量在 Gcc 中被视为弱符号。这背后的原因是什么?
最佳答案
gcc,C 模式:
未声明 extern
的未初始化全局变量被视为“通用”符号,而不是弱符号。
公共(public)符号在链接时合并,以便它们都引用相同的存储;如果不止一个对象试图初始化这样一个符号,您将得到一个链接时错误。 (如果它们没有在任何地方显式初始化,它们将被放置在 BSS 中,即初始化为 0。)
gcc,在 C++ 模式下:
不一样 - 它不做常见符号的事情。未声明为 extern
的“未初始化”全局变量被隐式初始化为默认值(简单类型或默认构造函数为 0)。
无论哪种情况,弱符号都允许初始化符号在链接时被同名的非弱初始化符号覆盖。
为了说明(这里集中在 C 案例上),我将使用一个主程序的 4 个变体,除了声明 global
的方式外,它们都是相同的:
main_init.c:
#include <stdio.h>
int global = 999;
int main(void) { printf("%d\n", global); return 0; }
main_uninit.c,省略了初始化:
#include <stdio.h>
int global;
int main(void) { printf("%d\n", global); return 0; }
main_uninit_extern.c,其中添加了extern
关键字:
#include <stdio.h>
extern int global;
int main(void) { printf("%d\n", global); return 0; }
main_init_weak.c,初始化 global
并将其声明为弱符号:
#include <stdio.h>
int global __attribute__((weak)) = 999;
int main(void) { printf("%d\n", global); return 0; }
和初始化相同全局的another_def.c:
int global = 1234;
单独使用 main_uninit.c
得到 0:
$ gcc -o test main_uninit.c && ./test
0
但是当 another_def.c
也包含在内时,global
被显式初始化,我们得到了预期的结果:
$ gcc -o test main_uninit.c another_def.c && ./test
1234
(请注意,如果您使用的是 C++,则这种情况会失败。)
如果我们尝试同时使用 main_init.c
和 another.def.c
,我们有 2 个 global
初始化,它们不会工作:
$ gcc -o test main_init.c another_def.c && ./test
/tmp/cc5DQeaz.o:(.data+0x0): multiple definition of `global'
/tmp/ccgyz6rL.o:(.data+0x0): first defined here
collect2: ld returned 1 exit status
main_uninit_extern.c
本身根本不起作用 - extern
关键字导致符号成为普通的外部引用而不是公共(public)符号,因此链接器提示:
$ gcc -o test main_uninit_extern.c && ./test
/tmp/ccqdYUIr.o: In function `main':
main_uninit_extern.c:(.text+0x12): undefined reference to `global'
collect2: ld returned 1 exit status
一旦包含来自 another_def.c
的初始化,它就可以正常工作:
$ gcc -o test main_uninit_extern.c another_def.c && ./test
1234
单独使用 main_init_weak.c
给出我们将弱符号初始化为 (999) 的值,因为没有任何东西可以覆盖它:
$ gcc -o test main_init_weak.c && ./test
999
但是从 another_def.c
中引入另一个定义在这种情况下确实有效,因为那里的强定义覆盖了 main_init_weak.c
中的弱定义:
$ gcc -o test main_init_weak.c another_def.c && ./test
1234
关于c++ - 为什么未初始化的全局变量是弱符号?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3691835/
我一直在尝试编写我自己的弱/强指针,但我并不清楚其中的关系。我似乎遇到的所有事情都没有说清楚,而且一个医生经常会反驳另一个医生所说的话。任何人都可以详细解释弱/强指针关系,也许还有图像或代码示例吗?
静态/动态和强/弱类型之间有什么区别? 最佳答案 静态/动态类型涉及何时获取类型信息(在编译时或运行时) 强/弱类型是关于如何严格区分类型(例如,语言是否尝试从字符串到数字进行隐式转换)。 请参阅wi
我有一个非常奇怪的情况。我的服务器当前已关闭并收到 503 http 状态代码。基于如下给定的代码,代码进入 if 条件,但是当我将调试点置于 let error = self?.decodeErro
对于短期运行的操作,避免[weak self]是否可以接受?例如,URLSession 将保留 dataTask(with:completion:) 的闭包: final class ViewCont
我有一个非常奇怪的情况。我的服务器当前已关闭并收到 503 http 状态代码。基于如下给定的代码,代码进入 if 条件,但是当我将调试点置于 let error = self?.decodeErro
假设我有以下情况: Test1.java import java.lang.ref.WeakReference; public class Test1 { public WeakReferen
有没有办法告诉模拟器(我正在使用 Modelsim)当信号不是由任一双向接口(interface)驱动时将信号拉到弱“H”? 例如,如果我有一个 I2C 信号 I2C_SDA 被声明为来自 2 个模块
这是将一些值放入 WeakHashMap 中然后从映射中删除这些值的代码片段。它如何处理分配的内存? import java.util.*; public class WeakHashMap_Main
我正在尝试弄清楚智能指针可以实现什么。 但有一些感觉像是障碍。 普通指针有一个简短的定义 Someclass *p但是智能指针有点长shared_ptr p当您必须处理这些指针的模板(如 vector
这两行代码有区别吗? __weak IBOutlet UITextField *usernameField; @property (weak) IBOutlet UITextField *userna
我最近发现了 WeakHashMap Java 中的数据结构。 但是,我不明白它在不再正常使用时对映射进行垃圾收集是什么意思。数据结构如何知道我将不再在我的程序中使用 key ?如果长时间不引用 ke
我的问题是为什么 weak IBOutletCollection 总是 nil?如果将弱变强,我所有的按钮都在那里,这真的很奇怪。我试图理解苹果的逻辑,我看不出单个按钮和一组按钮在内存管理方面没有区别
我创建一个 WeakHashMap 为 WeakHashMap map = new WeakHashMap(); map.put(emp,"hello"); 其中 emp 是一个 Employee 对
在delphi sydney中,在对象(不是接口(interface))前面设置[weak]会受到惩罚吗?示例: TMyObject = class(Tobject) Private
在delphi sydney中,在对象(不是接口(interface))前面设置[weak]会受到惩罚吗?示例: TMyObject = class(Tobject) Private
众所周知,我们将声明一个可以打破强引用循环的弱委托(delegate)对象: // MyObject.h ... @property (nonatomic, weak) id delegate; ..
我已阅读this article关于Java中不同类型的引用(强引用、软引用、弱引用、幻像引用),但我不太理解。 这些引用类型之间有什么区别?每种类型何时使用? 最佳答案 Java 提供了两种不同类型
我突然想到...我相信弱引用的生命 与该引用的范围(在函数内或全局内)相关。 所以我想知道,只要我将数据处理保持在特定范围内,那么我应该可以使用 weak 与 strong 引用。正确的? 我问的原因
func addAdditionalElement(_ additionalSelectedElementsIDs: [String], startX: CGFloat, containerView:
我想要一个指针,以便我可以判断引用计数何时为 1。本质上,指针的工作方式类似于 weak_ptr,但清理工作需要手动进行。也就是说,程序每隔一段时间就会经历一个指针循环,并检查哪些指针只剩下一个引用。
我是一名优秀的程序员,十分优秀!