C++对象内存模型1(堆栈模型)(转)
对象内存模型
一. 栈(Stack) VS. 堆(heap)
- 栈
- 由系统自动管理,以执行函数为单位
- 空间大小编译时确定(参数+局部变量)
- 函数执行时,系统自动分配一个stack
- 函数执行结束时,系统立即自动回收stack
- 堆
- 在c++中由程序员手动控制
- 手动分配new和malloc
- 手动释放delete和free
- 具有全局性,总体无大小限制
- 容易造成内存泄露
1. Myclass c(10); // 栈对象,空间大小在编译时确定,函数执行结束,系统立即回收
2.
Myclass* func(){
Myclass c();
return &c; //返回栈内存地址,指针悬浮,极端错误!
}
3.
Myclass func(){
Myclass c();
Aclass a();
c.pa = &a; //指向栈(local)对象,错!
return c;
}
结论: 指针指向栈对象,就要极度小心,一般会有问题!
4.
Myclass* func(){
Myclass *pa = new Myclass();
return pa; //小范围看没问题,但违背谁分配谁释放原则,此例接受者不知道要delete
}
总结返回指针问题:
返回栈指针: 完全错误
返回堆指针:释放可能有问题 ,易造成内存泄露
故一般不推荐
5 堆对象内存模型

6. 栈对象内存模型

二. 变量模型与使用 (对象,指针,引用; 声明,传参,返回值)
1. 声明与符号
“*” : 声明时,指针
用在指针前,解指针
“&”: 声明时,引用
用在对象前,取地址
MyClass c; //对象,一定在栈上 MyClass* pc; //指针,要问自己是栈指针,还是堆指针 MyClass& c2 = c; //引用,要问自己是栈引用,还是堆引用。本例为栈引用
// 举例堆引用 Myclass *pc2 = new Myclass(); Myclass& c3 = *pc2; c3为堆引用 c = *pc; //解指针, 可指向堆对象,也可以指向栈对象
//pc指向哪里,*pc即为指向哪里的对象(堆/栈)
//又c在栈上 再调用拷贝构造函数完成 c = *pc pc = &c; //取地址
2 传参分析
//对象
void func1(MyClass c) { // 对象往往较大,开销大,一般不!
}
//指针
void func2(MyClass* pc) { // 成本ok,但无法区分堆/栈,有delete问题
}
//引用
void func3(MyClass& mc) { //推荐,不想双向传递用const
}
const& 为 pass by value好的替代品
void func3(const MyClass& mc) {
}
调用方法:
MyClass c1;
func1(c1); //调用拷贝构造
func2(&c1); //不调用拷贝构造
func3(c1); //不调用拷贝构造,注意参数就写值即可,参考笔记2,传递者无需知道接受者的接受方式。乱加符号可能变成取地址了
3. 返回值分析
返回对象分析:
MyClass func1() {
MyClass c1;
return c1; //正确,调用拷贝构造
MyClass* pc2 = new MyClass();
return *pc2; // 返回时调用拷贝构造,结束后pc2就取不到了,内存无法释放,一定存在内存泄露
}
返回指针分析:
MyClass* func2() {
MyClass c1;
return &c1; //极端错误,返回栈对象指针
MyClass* pc2 = new MyClass();
return pc2; //不推荐,可能内存泄露,违背谁调用谁释放原则
}
故返回指针一般不推荐
返回引用分析:
MyClass& func3() {
MyClass c1;
return c1; // 极端错误,栈对象结束就消亡了
MyClass* pc2 = new MyClass();
return *pc2; //可能存在内存泄露,有办法取到pc2,但一般人不会去做,也不知道要做
}
返回传入参数的引用,ok且推荐,常见还有this指针
MyClass& func4(MyClass& c) {
return c;
}
C++对象内存模型1(堆栈模型)(转)的更多相关文章
- JVM 系列(4)一看就懂的对象内存布局
请点赞关注,你的支持对我意义重大. Hi,我是小彭.本文已收录到 GitHub · AndroidFamily 中.这里有 Android 进阶成长知识体系,有志同道合的朋友,关注公众号 [彭旭锐] ...
- C++/C#中堆栈、对象内存模型、深浅拷贝、Array.Clone方法
转载自:http://blog.csdn.net/jarvischu/article/details/6425534 目录 1. C++/C#中对象内存模型................. ...
- C++对象内存模型1(堆栈模型)
对象内存模型 一. 栈(Stack) VS. 堆(heap) 栈 由系统自动管理,以执行函数为单位 空间大小编译时确定(参数+局部变量) 函数执行时,系统自动分配一个stack 函数执行结束时,系统立 ...
- C#的对象内存模型
转载自:http://www.cnblogs.com/alana/archive/2012/07/05/2577893.html C#的对象内存模型: 一.栈内存和堆内存1.栈内存 由编译器自动分配和 ...
- C++对象内存模型2 (虚函数,虚指针,虚函数表)
从例子入手,考察如下带有虚函数的类的对象内存模型: class A { public: virtual void vfunc1(); virtual void vfunc2(); void func1 ...
- (转)c#对象内存模型
对象内存模型 C#的对象内存模型写这篇博客的主要目的是为了加深自己的理解,如有不对的地方,请各位见谅. C#的对象内存模型: 一.栈内存和堆内存1.栈内存 由编译器自动分配和释放,主要用来保存一些局部 ...
- 从零开始学C++之虚继承和虚函数对C++对象内存模型造成的影响
首先重新回顾一下关于类/对象大小的计算原则: 类大小计算遵循结构体对齐原则 第一个数据成员放在offset为0的位置 其它成员对齐至min(sizeof(member),#pragma pack(n) ...
- Swift 对象内存模型探究(一)
本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/zIkB9KnAt1YPWGOOwyqY3Q 作者:王 ...
- 对C++对象内存模型造成的影响(类/对象的大小)
首先重新回顾一下关于类/对象大小的计算原则: 类大小计算遵循结构体对齐原则 第一个数据成员放在offset为0的位置 其它成员对齐至min(sizeof(member),#pragma pack(n) ...
随机推荐
- go switch 和java C#有不同
1 switch 后的语句可以有简单的赋值语句 2 case :后的语句结束后不需要break;默认自动结束 除非以 fallthrough 语句结束,否则分支会自动终止 没有条件的 switch 有 ...
- go编译运行说明
二 编译运行说明 1.1 编译 1)有了go源文件,通过编译器将其编译成机器可以识别的二进制码文件. 2)在该源文件目录下,通过 go build 对hello.go 文件进行编译.可以指定生成的可 ...
- pytest_03_pycharm运行pytest (转:上海悠悠)
前言 上一篇pytest文档2-用例运行规则已经介绍了如何在cmd执行pytest用例,平常我们写代码在pycharm比较多 写完用例之后,需要调试看看,是不是能正常运行,如果每次跑去cmd执行,太麻 ...
- Bitnami配置域名访问
安装完成Bitnami后,需要执行以下命令将默认目录改为/wordpress: E:\Bitnami\wordpress-5.2.2-0\apps\wordpress\bnconfig.exe --a ...
- netcore访问本地磁盘
public void ConfigureServices(IServiceCollection services) { services.AddDirectoryBrowser(); }public ...
- oracle 逻辑对象
视图 视图的作用: a 让查询返回的字段更容易理解. b 对复杂select语句的封装. c 安全性.视图可以单独授权.对于数据量巨大的表,可以创建视图仅仅返回前100条数据,将该视图select权限 ...
- Java知识回顾 (17)MySQL链接
本部分介绍如何使用JDBC 连接 MySQL 数据库. 驱动包下载 Java 连接 MySQL 需要驱动包,最新版下载地址为:http://dev.mysql.com/downloads/connec ...
- S/4HANA Service Management和SAP Field Service Management的集成
经常有朋友提出这样的问题:"SAP Business Suite里的CRM的Service模块已经通过Addon的方式迁移到了S/4HANA上,并且SAP之前又收购了一家专门做Service ...
- dom4j 解析字符串成树形结构
引入maven依赖: <dependency> <groupId>dom4j</groupId> <artifactId>dom4j</artif ...
- python笔记--------一
作用域: 每个变量或函数都有自己的作用域. 每个函数都定义了一个命名空间,也称为作用域. 在最顶层有一个符号表会跟踪这一层所有的名称定义和和他们当前的绑定. 调用函数时,会建立一个新的符号表(常称为栈 ...