如果我有一个库foo,里面有一些静态初始化代码,并且链接情况如下:
executable -> libShared.so (dynamic linking)
executable -> libFoo.a (static linking)
libShared.so -> libFoo.a (static linking)这是否会导致静态初始化代码被运行两次-如果它有副作用,可能会做两个不同的可见事情?
libShared.so做了正确的事情,只导出了自己的符号,而没有导出任何libFoo.a。但是在这个过程中有两个libFoo.a副本,它们的名称空间实际上是分开的。我的理解对吗--会有一个重复的初始化步骤?
(当然,正确的方法是让libFoo.a不在其静态初始化器中创建可见的副作用,但让我们假设船已经启航了……)。
发布于 2017-07-27 05:48:26
是的,构造函数将运行两次。当您构建可执行或共享库时,会创建一个辅助方法,该方法枚举所有要构造的全局变量,并调用它们的初始化器(其他方法是ISO C++允许的,但不用于具有共享库的系统)。这些帮助器在加载可执行文件和共享库时调用。
现在恰好有两个这样的方法初始化内存的两个不同部分。这将创建两个对象,并且它们的地址将不同。
如果可执行文件只有一个全局的extern声明,情况就会有所不同,但显然它不能-它使用相同的静态库。
PS。全局对象的析构函数有自己的帮助器方法,工作方式大致相同。
发布于 2017-07-27 03:12:48
静态库只是目标文件的串联-其中没有代码来运行静态对象的构造函数,除非您显式地提供了这样的代码。因此,静态库中的静态对象不会让构造函数运行两次,甚至一次也不会。此外,由于大多数链接器搜索静态库的方式,因此多次链接它们是一种非常常见的做法。
https://stackoverflow.com/questions/45335499
复制相似问题