c++ 对象分配在栈上还是在堆上问题分析

 更新时间:2023年11月02日 11:24:07   作者:点墨  
这篇文章主要为大家介绍了c++ 对象在栈上还是在堆上问题分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

c++的对象到底在栈上还是分配在堆上?

首先,毋庸置疑,使用new和malloc系列函数分配的对象,一定是在堆上的。

Object *obj = new Object();

有争议的是

Object obj;

它是在栈上还是在堆上呢?

要回答这个问题,首先我们需要理解这句话的意思,这句话并不代表在栈上分配内存,它代表的是让obj具有“自动存储(automatic storage)”的性质。所谓的“自动存储”,意思是这个对象的存储位置取决于其声明所在的上下文。如果这个语句出现在函数内部,那么它就在栈上创建对象,此时obj变量和obj指代的对象(此时obj本质上其实是obj指代对象的首地址)都在栈上。

如果这个语句不是在函数内部,而是作为一个类的成员变量,则取决于这个类的对象是如何分配的。考虑下面的代码:

class Test{
    Object obj;
}
Test *test = new Test;

test指针是在栈上,它所指代的对象Test是存在堆上,那么obj变量和obj对象就在堆上。

class Test{
    Object obj;
}
Test test;

test变量在栈上,test对象在栈上,那么obj变量和obj对象就在栈上。

遵循这么一个原则:

指针变量和普通变量由上下文定义,指针所指向的内存在堆上,普通变量所指代的对象由上下文定义。

栈大小

栈大小是有默认值的,如果申请的临时变量太大就会超过栈大小,造成栈溢出。

它的默认值是可以修改的,一般,在unix-like平台,栈的大小是由环境变量控制的,所以不能通过设置编译器(像gcc)的任何编译标志来设置;在windows平台,栈的大小是包含在可执行文件里的,它可以在visual c++的编译过程中设置,但在gcc里是不可行的。

方法为:

项目->属性->链接器->系统->堆栈保留大小 (字节数)

在一般情况下,不同平台默认栈大小如下所示(仅供参考)

SunOS/Solaris 8172K bytes (Shared Version)
Linux 10240K bytes
Windows 1024K bytes (Release Version)
AIX 65536K bytes

演示

栈空间

代码

#include <iostream>
class Test {
public:
    Test() { std::cout << "Test" << std::endl; }
    ~Test() { std::cout << "~Test" << std::endl; }
private:
    char a[1024 * 1024];
};
class TestContainer {
public:
    TestContainer() { std::cout << "TestContainer" << std::endl; }
    ~TestContainer() { std::cout << "~TestContainer" << std::endl; }
private:
    Test test;
};
int main(int argc, char* argv[]) {
    TestContainer t;
    while (1) {}
    return 0;
}
  • 栈大小1MB(默认值),申请栈空间1MB

结果:程序崩溃,stack overflow

  • 栈大小1048577(1024*1024+1,即1MB多1B),申请栈空间1MB

结果:程序正常

堆空间

#include <iostream>
class Test {
public:
    Test() { std::cout << "Test" << std::endl; }
    ~Test() { std::cout << "~Test" << std::endl; }
private:
    char a[1024 * 1024];
};
class TestContainer {
public:
    TestContainer() { std::cout << "TestContainer" << std::endl; }
    ~TestContainer() { std::cout << "~TestContainer" << std::endl; }
private:
    Test test;
};
int main(int argc, char* argv[]) {
    TestContainer* t = new TestContainer;
    while (1) {}
    return 0;
}
  • 栈大小1MB,申请堆1MB
    结果:程序正常
  • 栈大小1MB,申请堆10MB
    结果:程序正常

以上就是c++ 对象分配在栈上还是在堆上问题分析的详细内容,更多关于c++ 对象堆栈分配的资料请关注脚本之家其它相关文章!

相关文章

  • c++11新增的便利算法实例分析

    c++11新增的便利算法实例分析

    这篇文章主要介绍了c++11新增的便利算法,主要有用于判断、查找、数组、序列等的操作算法,非常具有实用价值,需要的朋友可以参考下
    2014-09-09
  • C++中overload,override,overwrite的区别详细解析

    C++中overload,override,overwrite的区别详细解析

    以下是对C++中overload,override,overwrite的区别进行了详细的分析介绍,需要的朋友可以过来参考下
    2013-09-09
  • 简介C++编程中的运算符重载

    简介C++编程中的运算符重载

    这篇文章简单介绍了C++编程中的运算符重载,是C++入门学习中的基础知识,需要的朋友可以参考下
    2015-09-09
  • 纯C语言实现火车售票系统

    纯C语言实现火车售票系统

    这篇文章主要为大家详细介绍了纯C语言实现火车售票系统,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-12-12
  • C语言学生成绩管理系统设计

    C语言学生成绩管理系统设计

    这篇文章主要为大家详细介绍了C语言学生成绩管理系统设计,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-01-01
  • C语言求解无向图顶点之间的所有最短路径

    C语言求解无向图顶点之间的所有最短路径

    这篇文章主要为大家详细介绍了C语言求解无向图顶点之间的所有最短路径,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2019-01-01
  • MFC实现简单计算器

    MFC实现简单计算器

    这篇文章主要为大家详细介绍了MFC实现简单的计算器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-05-05
  • 详解C++11中绑定器bind的原理与使用

    详解C++11中绑定器bind的原理与使用

    C++11中引入的function机制,其中绑定器主要有三种:bind1st、bind2nd、bind(C++11)。本文就来和大家聊聊这些绑定器的底层实现原理与使用场景,需要的可以参考一下
    2022-12-12
  • C/C++中#define的妙用分享

    C/C++中#define的妙用分享

    本文主要介绍了C++/C关于#define的一些妙用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-02-02
  • C++类和对象补充

    C++类和对象补充

    类是创建对象的模板,一个类可以创建多个对象,每个对象都是类类型的一个变量;创建对象的过程也叫类的实例化。每个对象都是类的一个具体实例(Instance),拥有类的成员变量和成员函数
    2021-10-10

最新评论