好的,我找到了解决方案:
http://gcc.gnu.org/wiki/可见性
例如,如果更改
static CTest test;
至
__attribute__ ((visibility ('hidden'))) static CTest test;
问题将会消失。Linux:
Ctest() this=-1646158468Ctest() this=6296196useuse~Ctest() this=6296196, is Alive~Ctest() this=-1646158468, is Alive
修复前的nm输出为:
0000000000200dd4 B _ZN5CTest4testE
修复后:
0000000000200d7c b _ZN5CTest4testE
差异将全局符号“ B”更改为局部符号“ b”。
除了在符号上添加“ ((visibility(“ hidden”)))“之外,还可以使用编译器选项” -fvisibility =hidden“。该选项使gcc的行为更像Windows env。
解决方法环境:linux x64,编译器gcc 4.x
项目具有以下结构:
static library 'slib'-- inside this library,there is static object 'sobj'dynamic library 'dlib'-- links statically 'slib'executable 'exe':-- links 'slib' statically-- links 'dlib' dynamically
在程序结束时,“ sobj”被破坏两次。这种行为是可以预期的,但是在相同的内存地址(即在析构函数中使用相同的“this”)会被破坏两次,结果是存在双重破坏问题。我认为这是由于某些符号重叠造成的。
该冲突有什么解决方案?也许一些链接选项?
这是测试用例:
main_exe.cpp
#include <cstdlib>#include 'static_lib.h'#include 'dynamic_lib.h'int main(int argc,char *argv[]){ stat_useStatic(); din_useStatic(); return EXIT_SUCCESS;}
static_lib.h
#ifndef STATIC_LIB_H#define STATIC_LIB_H#include <cstdio>void stat_useStatic();struct CTest{ CTest(): status(isAlive) {printf('CTest() this=%dn',this); } ~CTest() {printf('~CTest() this=%d,%sn',this,status==isAlive?'is Alive':'is Dead');status=isDead; } void use() {printf('usen'); } static const int isAlive=12385423; static const int isDead=6543421; int status; static CTest test;};#endif
static_lib.cpp
#include 'static_lib.h'CTest CTest::test;void stat_useStatic(){ CTest::test.use();}
dynamic_lib.h
#ifndef DYNAMIC_LIB_H#define DYNAMIC_LIB_H#include 'static_lib.h'#ifdef WIN32#define DLLExport __declspec(dllexport)#else#define DLLExport #endifDLLExport void din_useStatic();#endif
dynamic_lib.cpp
#include 'dynamic_lib.h'DLLExport void din_useStatic(){ CTest::test.use();}
CMakeLists.txt
project( StaticProblem )cmake_minimum_required(VERSION 2.6)if(WIN32)else(WIN32) ADD_DEFINITIONS(-fPIC)endif(WIN32)ADD_LIBRARY( static_lib STATIC static_lib.cpp static_lib.h)ADD_LIBRARY( dynamic_lib SHARED dynamic_lib.cpp dynamic_lib.h)TARGET_LINK_LIBRARIES( dynamic_lib static_lib )ADD_EXECUTABLE( main_exe main_exe.cpp )TARGET_LINK_LIBRARIES( main_exe static_lib dynamic_lib )
该示例在Windows上可以正常运行,但是在Linux上-存在问题。因为它在Windows上可以正常工作,所以解决方案应该像更改某些链接选项或类似的操作,而不是更改项目结构或不使用静态var。
输出:
视窗
CTest() this=268472624CTest() this=4231488useuse~CTest() this=4231488,is Alive~CTest() this=268472624,is Alive
的Linux
CTest() this=6296204CTest() this=6296204useuse~CTest() this=6296204,is Alive~CTest() this=6296204,is Dead