答案是不会。这是测试的答案。
现在知道原理了,尽管这是windows via C/C++中解释的windows的做法,但是我想linux也是这么类似处理的。
windows 使用memory map来加载exe和dll。当一个exe/dll有多个instance要启动时,实际在windows paging file(包括RAM和swap文件)中,exe和牵涉到的dll,只有一份。这样可以节省内存使用,也可以提高性能。
也就是说,如果是exe,虽然每个instance都有自己独立的地址空间,但是地址空间映射到storage的时候,他们映射的都是同样的地方。这样就带来问题了:exe/dll中的全局变量和静态变量怎么办?每个instance都有可能会修改这些变量。
windows 的做法是,该存放全局变量和静态变量的page,设定copy on write protect attribute。所以,当任何一个线程尝试修改这些page中的内容时,windows负责分配一个新的page出来,然后修改该线程的地址空间,将 这个新分配的page的地址设置上去,从此以后,该线程修改这个全局变量或是静态变量, *** 作的就是这个新分配的page了。这样,多个实例就不会出现全局 变量或静态变量互相覆盖的问题了。
参考windows via C/C++ P593,有详细说明。
曾经在一次写程序的时候,全局使用一个日志对象,所有的日志信息都由此日志对象写入文件。但是发现在动态链接库里,无法存取到该全局对象,后经查找资料。发现,动态链接库是否能存取主程序的全局变量,主程序是否能存取动态链接库里定义的全局变量都是可以通过链接指令改变此行为。
主程序存取动态链接库里的全局变量
例如动态库里定义全局变量int i, 在主程序里申明extern int i。 则,主程序存取的就是动态库里定义的i。在所有的UNIX平台上这是默认的行为(注意:实际定义i的 模块的点o文件,必须同时链接到动态库和主程序上,否则链接报错)。
如果不想让主程序能存取动态库里的全局变量,则在链接动态连接库的时候,给gcc传入-Wl,-Bsymbolic即可。
动态连接库存取主程序里定义的全局变量
在linux上,链接主程序的时候,使用参数-Wl,–export-dynamic
在AIX上,使用’deferred imports’ 并且 enable ‘runtime linking’.
PS:在linux 下 加入-export -dynamic 可以实现
如果能有好的相关文章参考就更好了。------解决方案--------------------------------------------------------
必然有可重入性问题!
这个会有问题的哦, 属于重定义, 在动态库里的变量最好做成static的, 或者通过函数手段取地址跨文件引用避免暴漏在全局作用域.
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)