限制对象在堆或栈中声明

***********************************************声明************************************************************

原创作品,出自 “晓风残月xj” 博客,欢迎转载,转载时请务必注明出处(http://blog.csdn.net/xiaofengcanyuexj)。

由于各种原因,可能存在诸多不足,欢迎斧正!

****************************************************************************************************************

最近学习中,遇到一些平时很少使用的C++语言知识,其实扩展开去也不仅仅限于C++语言知识,很多面向对象的语言都会遇到类似的问题。今天结合资料做一下小结,也希望你能够帮助到有需要的朋友。

通常对象可以定义在对或栈中,一般用new,malloc等函数或运算符声明的对象定义在堆中,其他的定义在栈中。

栈区(stack):由编译器自动分配释放,存放函数的参数值,局部变量的值等,其组织方式类似于数据结构中的栈。

堆区(heap):一般由程序员分配释放,若程序员不释放,程序结束时可能由OS回收,其组织方式方式倒是类似于链表。关于程序中内存空间的分配机制参见这篇文章点击

栈上分配空间优点:快速高效,对象生命周期是自动的,离开作用域之后就自动析构回收。缺点是栈空间有限,而且不能人为控制对象的生存期,比如你无法将一个函数内部的栈上的对象返回,因为这个函数执行完毕后栈空间会自动回收;

堆上分配空间就相反,创建对象往往效率较低,且对堆对象的访问也稍慢于栈对象。一旦创建成功,需要自己手动delete回收,系统不会帮你管理回收(用GC的除外),但是分配空间大小灵活,而且正因为是手动回收的,你可以自由控制对象生存期,常用来跨域传递对象。

一般情况下对象在堆还是栈上声明没有过多的要求,但有时需要明确指定定义方式,甚至是2选1。C++的重载机制+访问控制机制,重载new运算符将其设成私有成员函数可以达到只在栈上分配的目的;C++的访问控制机制,将构造函数设为私有可以达到只在堆上分配的目的。

下面是一段辅助说明的源代码:

#include<iostream>
using namespace std;

class CHeapAndStack
{
public:
	CHeapAndStack()
	{
		cout<<"Contructed  CHeapAndStack!"<<endl;
	}
	~CHeapAndStack()
	{
		cout<<"Destructed  CHeapAndStack!"<<endl;
	}
};

class CHeapOnly
{
public:
	static CHeapOnly *CreateInstance()
	{
		return new CHeapOnly();
	}

private:
	CHeapOnly()
	{
		cout<<"Constructed  CHeapOnly!"<<endl;
	}
};

class CStackOnly
{
private:
	void * operator new(size_t size)
	{
		cout<<"Constructed  CStackOnly!"<<endl;
	}
};

int main()
{
	CHeapAndStack heapAndStackA;
	CHeapAndStack* heapAndStack=new  CHeapAndStack();

	//CHeapOnly heapOnlyA;//编译出错
	CHeapOnly *heapOnlyB = CHeapOnly::CreateInstance();

	CStackOnly StackOnlyA;
	//CStackOnly *StackOnlyB = new CStackOnly();//编译出错

	system("pause");
	return 0;
}
时间: 2024-10-31 13:42:22

限制对象在堆或栈中声明的相关文章

如何让对象只在堆或者栈中分配空间

在开始之前先来分析一下C++中的new运算符和operator new之间的关联. new:指我们在C++里通常用到的运算符,比如A* a = new A或者调用带参数的构造函数;  对于new来说,有new和::new之分,前者位于std. operator new():指对new运算符的重载形式,它是一个函数,并不是运算符.对于operator new来说,分为全局重载和类重载,全局重载是void* ::operator new(size_t size),在类中重载形式 void* A::o

Java中对象、对象引用、堆、栈、值传递以及引用传递的详细解释

Java中对象.对象引用.堆.栈.值传递以及引用传递的详细解释 1.对象和对象引用的区别: (1).对象: 万物皆对象.对象是类的实例.在Java中new是用来在堆上创建对象用的.一个对象可以被多个引用所指向. (2).对象引用: 类似于C++中的地址.通过对象引用可以找到对象.一个引用可以指向多个对象.操纵的标识符实际上是指向对象的引用. 就像:对象存放在屋子里,对象的引用就相当于屋子的钥匙. 2.值传递和引用传递的区别: (1).值传递:传递的是值的拷贝.也就是说传递后就不互相关了. (2)

内存分配中的堆、栈、静态区、只读区

内存中的栈区处于相对较高的地址以地址的增长方向为上的话,栈地址是向下增长的,栈中分配局部变量空间,堆区是向上增长的用于分配程序员申请的内存空间.另外还有静态区是分配静态变量,全局变量空间的:只读区是分配常量和程序代码空间的:以及其他一些分区.对于常量,在实际情况中,是会复用的,比如变量a和b都赋值为”abc”则实际上他们指向同一块地址. main.cpp int a = 0; //全局初始化区 char *p1; //全局未初始化区 main() { int b; //栈 char s[] =

关于内存中栈和堆的区别(非数据结构中的堆和栈,区别)

本文转载自:http://blog.csdn.net/fenghuayoushi/article/details/6074666 ----------------------------------------------------------------------------------------------------------------------- 下面内容转自:http://www.cnblogs.com/gesenkof99/archive/2009/04/14/14356

数据结构中的堆和栈 与 内存分配中的堆区和栈区 分析

比較全面的总结了诸多版本号,知识无国界.感谢各位的辛勤劳作. 在计算机领域,堆栈是一个不容忽视的概念,我们编写的C/C++语言程序基本上都要用到.但对于非常多的初学着来说,堆栈是一个非常模糊的概念. (1) 数据结构的栈和堆 首先在数据结构上要知道堆栈,虽然我们这么称呼它,但实际上堆栈是两种数据结构:堆和栈. 堆和栈都是一种数据项按序排列的数据结构. 栈就像装数据的桶或箱子 我们先从大家比較熟悉的栈说起吧.它是一种具有后进先出性质的数据结构,也就是说后存放的先取.先存放的后取.这就如同我们要取出

C语言中堆和栈的区别

原文:http://blog.csdn.net/tigerjibo/article/details/7423728 一.前言: C语言程序经过编译连接后形成编译.连接后形成的二进制映像文件由栈,堆,数据段(由三部分部分组成:只读数据段,已经初始化读写数据段,未初始化数据段即BBS)和代码段组成,如下图所示: 1.栈区(stack):由编译器自动分配释放,存放函数的参数值,局部变量等值.其操作方式类似于数据结构中的栈. 2.堆区(heap):一般由程序员分配释放,若程序员不释放,则可能会引起内存泄

关于内存中堆和栈的知识和应用的总结

以前做的一个小项目最近经常不稳定,查了下日志大部分都是因为缓冲溢出而导致程序报错,于是乎在网上查找了一些关于内存使用的内容,收获颇丰.在此将前人的一些精华总结收录于此,加深印象.有不对或不足的地方请大家予以指正. 文章链接:http://blog.csdn.net/szchtx/article/details/7981401 http://www.cppblog.com/oosky/archive/2006/01/21/2958.html http://www.cnblogs.com/lln77

C++中堆和栈的完全解析

内存分配方面: 堆: 操作系统有一个记录空闲内存地址的链表,当系统收到程序的申请时,会遍历该链表,寻找第一个空间大于所申请空间的堆结点,然后将该结点从空闲结点链表中删 除,并将该结点的空间分配给程序,另外,对于大多数系统,会在这块内存空间中的首地址处记录本次分配的大小,这样代码 中的delete语句才能正确的释放本内存空间.我们常说的内存泄露,最常见的就是堆泄露(还有资源泄露),它是指程序在运行中出现泄露,如果程序被关闭掉的话,操作系统会帮助释放泄露的内存. 栈:在函数调用时第一个进栈的主函数中

C语言中的堆与栈20160604

首先声明这里说的是C语言中的堆与栈,并不是数据结构中的!一.前言介绍:C语言程序经过编译连接后形成编译.连接后形成的二进制映像文件由栈,堆,数据段(由三部分部分组成:只读数据 段,已经初始化读写数据段,未初始化数据段即BBS)和代码段组成,如下图所示: 其中所谓静态,就是一定会存在的而且会永恒存在.不会消失,这样的数据包括常量.常变量(const 变量).静态变 量.全局变量等. 动态的话,就是会变化的了.动态的区域,就是堆和栈.这个栈应该称作 call stack,上面会存放函数的返回地址.