malloc.c:3097: sYSMALLOc:断言在动态创建的对象的构造函数中失败

malloc.c:3097: sYSMALLOc: Assertion fails inside constructor of dynamically created object

本文关键字:对象 构造函数 失败 创建 3097 sYSMALLOc 断言 malloc 动态      更新时间:2023-10-16

我对 OOP C++有点陌生,所以我希望我只是犯了一些简单的初学者错误。我有一些代码给了我看似莫名其妙的malloc错误。我已经将我的代码简化为仍然崩溃的最小内容,即:

我的类.cpp:

#include <vector>
using std::vector;
#include <string>
using std::string;
class MyClass{
      string * myString1;
      string * myString2;
      int length1;
      int length2;
      vector< vector<int>* > * myVector;
   public:
      MyClass(string * a, string * b);
};
MyClass::MyClass(string * a, string * b)
: myString1(a), myString2(b)
{
   // commenting both of the lines that access the string sizes
   // makes it work
   length1 = myString1->size();
   // commenting either or both of the following lines makes
   // it work
   length2 = myString2->size();
   myVector = new vector< vector<int>* >;
}

MyClass.h

#ifndef MyClass_H
#define MyClass_H
#include <string>
using std::string;
class MyClass{
   public:
      MyClass(string * a, string * b);
};
#endif

测试.cpp:

#include "../include/MyClass.h"
#include <string>
using std::string;

int main()
{
   string * A = new string("foobar");
   string * B = new string("foobaz");
   MyClass * mc = new MyClass(A, B);
   return 0;
}

我像这样编译:

g++ -ggdb -Wall -c src/test.cpp -o .objects/test.o
g++ -ggdb -Wall -c src/MyClass.cpp -o .objects/MyClass.o
g++ .objects/test.o .objects/MyClass.o  -ggdb -Wall -o test

当我运行它时,我得到这个:

test: malloc.c:3097: sYSMALLOc: Assertion `(old_top == (((mbinptr) (((char *) &((av)->bins[((1) - 1) * 2])) - __builtin_offsetof (struct malloc_chunk, fd)))) && old_size == 0) || ((unsigned long) (old_size) >= (unsigned long)((((__builtin_offsetof (struct malloc_chunk, fd_nextsize))+((2 * (sizeof(size_t))) - 1)) & ~((2 * (sizeof(size_t))) - 1))) && ((old_top)->size & 0x1) && ((unsigned long)old_end & pagemask) == 0)' failed.
Aborted

我已经看到 valgrind 被建议作为调试此类问题的工具,但我无法从它的输出中理解任何东西。输出很长,所以我把它粘贴在这里。

Gdb 指出创建新向量的行是问题所在(这并不太令人惊讶(。完整的 gdb 输出在这里。我觉得奇怪的是,如果我不访问任何一个字符串长度,那么新向量就会毫无问题地创建。

我在这里真的很茫然,任何帮助将不胜感激!

主要问题是你以两种不兼容的方式定义了MyClass:在MyClass.cpp中,它有五个数据成员(两个指针、两个整数和另一个指针(,但在MyClass.h中,因此在test.cpp中,它根本没有数据成员。因此,当您的main函数调用new MyClass(A, B)时,系统只为MyClass的无数据成员版本分配足够的内存,然后调用构造函数,该构造函数初始化您未为其分配内存的数据成员。

你需要将类定义从MyClass.cpp移动到MyClass.h,并且MyClass.cpp的第一行应该是#include "MyClass.h"。这将确保您没有任何不兼容,因为如果您这样做,MyClass.cpp将导致 发出编译错误。

你已经声明了两个完全不同的MyClass版本;一个在MyClass.h,另一个在MyClass.cpptest.cpp 中的 main() 函数会看到 MyClass.h 中的版本,并根据该定义分配内存,但 MyClass.cpp 中的代码正在处理该类的不同定义。

失败的原因是这两个定义在内存中具有不同的大小和布局。

要正确地执行此操作,您需要在MyClass.h中包含MyClass的完整定义。即

#ifndef MyClass_H
#define MyClass_H
#include <string>
using std::string;
class MyClass {
      string * myString1;
      string * myString2;
      int length1;
      int length2;
      vector< vector<int>* > * myVector;
   public:
      MyClass(string * a, string * b);
};
#endif

这是C++的关键点之一 - 即使是私有成员也需要在可公开访问的类定义中。

然后你需要在MyClass.cpptest.cpp中包含MyClass.h,并从MyClass.cpp中删除MyClass声明——你只希望类声明出现在一个地方。


还有内存泄漏,因为您没有MyClass析构函数提供析构函数,它将在其中清理在其构造函数中分配的内存。到处写newdelete的更好方法是调查std::shared_ptr。