指针神秘地重置为NULL

时间:2009-03-02 00:31:21

标签: c++ pointers

我正在开发一款游戏,而我正在研究处理输入的部分。这里涉及三个类,其中ProjectInstance类启动级别和东西,有一个GameController将处理输入,一个PlayerEntity将受到控件的影响由GameController。在启动级别时,ProjectInstance创建GameController,它将在Step方法中调用其EvaluateControls方法,该方法在游戏循环内调用。 EvaluateControls方法看起来有点像这样:

void CGameController::EvaluateControls(CInputBindings *pib) {
    // if no player yet
    if (gc_ppePlayer == NULL) {
        // create it
        Handle<CPlayerEntityProperties> hep = memNew(CPlayerEntityProperties);
        gc_ppePlayer = (CPlayerEntity *)hep->SpawnEntity();
        memDelete((CPlayerEntityProperties *)hep);
        ASSERT(gc_ppePlayer != NULL);
        return;
    }

    // handles controls here
}

正确调用此函数,并且assert永远不会触发。但是,每次调用此函数时,gc_ppePlayer都设置为NULL。正如您所看到的,它不是超出范围的局部变量。唯一的地方gc_ppePlayer可以设置为NULL在构造函数中或可能在析构函数中,在EvaluateControls的调用之间都没有调用它们。调试时,gc_ppePlayer在返回之前会收到正确的预期值。当我再次按F10并且光标位于右括号时,值将更改为0xffffffff。我在这里不知所措,怎么会发生这种情况?任何人吗?

5 个答案:

答案 0 :(得分:5)

当该表达式的值发生变化(为NULL或从NULL)时,在gc_ppePlayer == NULL上设置一个观察点,调试器将指向您确切的位置。

试试看看会发生什么。寻找未终止的字符串或mempcy复制到太小的内存等...通常这是导致全局/堆栈变量被随机覆盖的原因。

在VS2005中添加观察点(brone说明)

  
      
  1. 转到“断点”窗口
  2.   
  3. 点击新建,
  4.   
  5. 单击“数据断点”。输入
  6.   地址栏中的
  7. &gc_ppePlayer,请离开   其他价值观。
  8.   
  9. 然后跑。
  10.         

    gc_ppePlayer发生变化时   断点       会受到打击。 - 布鲁恩

答案 1 :(得分:2)

您是在调试Release还是Debug配置?在发布版本配置中,您在调试器中看到的并不总是如此。进行了优化,这可以使观察窗口显示出您所看到的奇特值。

你真的看到了ASSERT触发吗? ASSERT通常是在发布版本中编译的,所以我猜你正在调试发布版本,这就是为什么ASSERT不会导致应用程序终止。

我建议构建该软件的Debug版本,然后查看gc_ppePlayer是否真的为NULL。如果真的如此,也许你会看到某种类型的内存堆损坏,其中该指针被覆盖。但是如果它是内存损坏,它通常会比你描述的更不确定。

顺便说一句,使用像这样的全局指针值通常被认为是不好的做法。如果它是真正的单个对象并且需要全局可访问,请查看是否可以用单例类替换它。

答案 2 :(得分:2)

我的第一个想法是说SpawnEntity()返回一个指向内部成员的指针,该成员在调用memDelete()时被“清除”。当指针设置为0xffffffff时,我不清楚,但如果它在调用memDelete()期间发生,那么这就解释了为什么你的ASSERT没有触发 - 0xffffffff与NULL不同。

自从重建整个代码库以来已经有多长时间了?我一次又一次地看到这样的记忆问题,只需重建整个解决方案就可以解决这个问题。

您是否尝试过(F11)而不是功能结束时的步骤(F10)?虽然您的示例没有显示任何局部变量,但为了简单起见,您可能会留下一些变量。如果是这样,F11将(希望)进入任何这些变量的析构函数,允许您查看其中一个是否导致问题。

答案 3 :(得分:0)

你有一个“fandango on core。”

动态初始化会覆盖内存的各种位(原文如此)。

直接或间接地覆盖了全局。 相对于堆的内存中的全局在哪里?

二进制切断动态初始化部分直到问题消失。 (一次一半地评论出来,递归)

答案 4 :(得分:0)

根据您所使用的平台,可以快速找出此类内存问题的工具(免费或付费)。

脱离我的头顶: