何时使用 alloca 为类成员分配内存?

Posted

技术标签:

【中文标题】何时使用 alloca 为类成员分配内存?【英文标题】:When is memory allocated using alloca freed for class members? 【发布时间】:2011-12-20 05:51:16 【问题描述】:
class MyString

  public:
  MyString(int length):_ptr(alloca(length))
  
  
  //Copy Constructor, destructor, other member functions.
  private:
  void* _ptr;
;

int main()

  MyString str(44);
  return 0;

它是在主函数结束时释放还是在构造函数执行后立即释放? 如果上面的代码按预期工作,那么有一个这样的字符串类是个好主意吗?

更新:

看起来主要的危险是

    堆栈溢出 构造函数的内联

我认为我可以通过将 alloca 用于小尺寸并使用 malloc/free 用于大尺寸来处理 ***。我想必须有一些非便携式编译器特定的方式来强制编译器内联。

我很感兴趣,因为字符串类在任何 c++ 项目中都被广泛使用。如果我做对了,我预计会有巨大的性能提升,因为大多数分配都进入堆栈,否则会进入堆。这将是一个实用程序,最终用户不会知道内部结构。

【问题讨论】:

这取决于您选择的优化级别。它仅在构造函数被内联时才有效。不要使用这个。 【参考方案1】:

根据alloca的文档,当alloca的调用者返回时,内存被释放。因此,如果初始化列表被编译器视为构造函数的一部分,则内存将在构造函数的末尾被释放,即堆栈帧是在初始化列表执行之前创建的。如果在执行初始化列表后创建堆栈帧,则分配的内存将在构造函数的调用者中,因此,在这种情况下,内存在 main 结束时被释放。我对标准的了解还不够,无法绝对确定它会以哪种方式发生。

但是,无论何时最终释放内存,ptr 的值都不会改变。

【讨论】:

+1 我相信 initializer list 是在构造函数的堆栈框架中处理的,因为它可能在不同的翻译单元中,编译器不可能知道它有分配额外的空间之前调用构造函数。 @David Rodríguez - dribeas:绝对。至少从实现的角度来看,初始化器的构造代码是在父 c'tor 的范围内执行的【参考方案2】:

类成员的初始化肯定是作为 c'tor 的一部分执行的。因此,至少按照标准,alloca 返回的指针的有效性仅限于 c'tor。

因此,以您的方式初始化您的班级成员似乎是一个非常糟糕的主意。

OTOH 不会有以下问题:

class MyString

  public:
  MyString(void* ptr):_ptr(ptr)
  
  
  //Copy Constructor, destructor, other member functions.
  private:
  void* _ptr;
;

int main()

  MyString str(alloca(44));
  return 0;

【讨论】:

以上是关于何时使用 alloca 为类成员分配内存?的主要内容,如果未能解决你的问题,请参考以下文章

何时为 C++ 中的类的成员函数分配内存空间?

C alloca 函数 - 当试图分配太多内存时会发生啥

C:malloc/calloc/realloc/alloca内存分配函数

第十二章 类和动态内存分配

alloca() 可以替代 C++ 中的自动指针吗?

alloca() 如何在内存级别上工作?