当前位置: 首页 > 面试题库 >

C ++ Linux双重破坏了静态变量。链接符号重叠

麻茂材
2023-03-14
问题内容

环境: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=%d\n",this);
    }
    ~CTest()
    {
        printf("~CTest() this=%d, %s\n",this,status==isAlive?"is Alive":"is Dead");
        status=isDead;
    }
    void use()
    {
        printf("use\n");
    }
    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 
#endif
DLLExport 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=268472624
CTest() this=4231488
use
use
~CTest() this=4231488, is Alive
~CTest() this=268472624, is Alive

的Linux

CTest() this=6296204
CTest() this=6296204
use
use
~CTest() this=6296204, is Alive
~CTest() this=6296204, is Dead

问题答案:

好的,我找到了解决方案:

http://gcc.gnu.org/wiki/可见性

例如,如果更改

static CTest test;

__attribute__ ((visibility ("hidden"))) static CTest test;

问题将会消失。Linux:

CTest() this=-1646158468
CTest() this=6296196
use
use
~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。



 类似资料:
  • 问题内容: 我有一个数组,其中一个键包含一些HTML代码。我在json_encoding此数组以供返回。我最近注意到,如果html包含特殊字符(即:viraga),则json_encode()返回“ null”。 如果有特殊的重音字符,我如何才能使json_encode()不返回’null’? 问题答案: 假定文本采用UTF-8编码。如果输入看起来像格式不正确的UTF-8,则返回。上班的唯一方法是

  • 问题内容: 我习惯在所有程序中广泛使用 静态变量 ,尤其是在使用Android时。我倾向于使用它们,因为有时 通过Intents发送10个或更多的值 会很 麻烦 。因此,我只是将 它们声明为静态 变量,并通过使用“点”运算符轻松地在其他类中访问它们。使用静态变量的另一个原因是当我要在整个应用程序中使用 Utility类时 。就像我在下面给出的代码一样,它可以帮助我 在不同的活动中使用变量 。 Ut

  • 问题内容: 实际上,Linux上的 -static gcc标志现在不起作用。让我从GNU libc常见问题中引用: 2.22。 即使是静态链接程序也需要一些共享库,这对我来说是不可接受的。我能做什么? {AJ} NSS(有关详细信息,请键入“ info libc“名称服务开关”“)在没有共享库的情况下无法正常工作。NSS只需更改一个配置文件(/etc/nsswitch.conf)即可使用不同的服务

  • 帮我理解这个...参见粗体。从标准3.6.3终止(2) 如果一个函数包含一个静态或线程存储持续时间的块作用域对象,该对象已经被破坏,并且该函数在一个静态或线程存储持续时间的对象被破坏期间被调用,如果控制的信息通过了先前被破坏的blockscope对象的识别,则该程序具有未被修改的行为。同样,如果块作用域对象在销毁后被间接使用(即通过指针),则该行为将被忽略。 有人对此有明确的见解吗?

  • 问题内容: 我有以下类别,将其用作项目中所有模型的基础: 然后,我尝试通过以下方式对其进行扩展: 但是,如果我尝试执行以下操作: 然后,而不是查询:,它生成查询:“从空WHERE id =?中选择SELECT ID”。因此,覆盖类中的字段似乎没有任何效果。 我该如何克服呢?如果我在BaseModel中添加了一个方法,并在的构造函数中对其进行了调用,那么该类的所有方法都可以使用的新值吗? 问题答案:

  • 在我的VC++2008项目中,我创建了两个文件globals.h和globals.cpp,如下所示: glboals.h: globals.cpp: 然后我在另一个文件testglboaldlg.cpp中使用全局变量,如下所示: 但得到以下链接错误: