C++ 基础6 内存与引用_HuiFeiDeTuoNiaoGZ的博客-程序员秘密

技术标签: C++  

内存分区模型

C++程序在执行时,将内存大方向划分为4个区域

代码区:存放函数体的二进制代码,由操作系统进行管理的
全局区:存放全局变量和静态变量以及常量
栈区:由编译器自动分配释放, 存放函数的参数值,局部变量等
堆区:由程序员分配和释放,若程序员不释放,程序结束时由操作系统回收

意义
不同区域存放的数据,赋予不同的生命周期, 给我们更大的灵活编程

程序运行之前

在程序编译后,生成了exe可执行程序,未执行该程序前分为两个区域

​ 代码区:

​ 存放 CPU 执行的机器指令

​ 代码区是共享的,共享的目的是对于频繁被执行的程序,只需要在内存中有一份代码即可

​ 代码区是只读的,使其只读的原因是防止程序意外地修改了它的指令

​ 全局区:

​ 全局变量和静态变量存放在此.

​ 全局区还包含了常量区, 字符串常量和其他常量也存放在此.

​ 该区域的数据在程序结束后由操作系统释放.

//全局变量
int g_a = 10;
int g_b = 10;

//全局常量
const int c_g_a = 10;
const int c_g_b = 10;

int main() {
    

	//局部变量
	int a = 10;
	int b = 10;

	//打印地址
	cout << "局部变量a地址为: " << (int)&a << endl;
	cout << "局部变量b地址为: " << (int)&b << endl;

	cout << "全局变量g_a地址为: " <<  (int)&g_a << endl;
	cout << "全局变量g_b地址为: " <<  (int)&g_b << endl;

	//静态变量
	static int s_a = 10;
	static int s_b = 10;

	cout << "静态变量s_a地址为: " << (int)&s_a << endl;
	cout << "静态变量s_b地址为: " << (int)&s_b << endl;

	cout << "字符串常量地址为: " << (int)&"hello world" << endl;
	cout << "字符串常量地址为: " << (int)&"hello world1" << endl;

	cout << "全局常量c_g_a地址为: " << (int)&c_g_a << endl;
	cout << "全局常量c_g_b地址为: " << (int)&c_g_b << endl;

	const int c_l_a = 10;
	const int c_l_b = 10;
	cout << "局部常量c_l_a地址为: " << (int)&c_l_a << endl;
	cout << "局部常量c_l_b地址为: " << (int)&c_l_b << endl;

	system("pause");

	return 0;
}

C++中在程序运行前分为全局区和代码区
代码区特点是共享和只读
全局区中存放全局变量、静态变量、常量
常量区中存放 const修饰的全局常量 和 字符串常量

程序运行之后

栈区:

​ 由编译器自动分配释放, 存放函数的参数值,局部变量等

​ 注意事项:不要返回局部变量的地址,栈区开辟的数据由编译器自动释放
意思是局部变量在其生命周期外的地址不可用

int * func()
{
    
	int a = 10;
	return &a;
}

int main() {
    

	int *p = func();

	cout << *p << endl;//10,因为编译器在func()执行完后为了保险起见多保留了一次局部变量的值
	cout << *p << endl;//乱码,此时a已被释放

	system("pause");

	return 0;
}

堆区:

​ 由程序员分配释放,若程序员不释放,程序结束时由操作系统回收

​ 在C++中主要利用new在堆区开辟内存

int* func()
{
    
	int* a = new int(10);
	return a;
}

int main() {
    

	int *p = func();

	cout << *p << endl;//10
	cout << *p << endl;//10,堆区中的变量(new int)不会被自动释放,除非手动操作
    
	system("pause");

	return 0;
}

总结:

堆区数据由程序员管理开辟和释放

堆区数据利用new关键字进行开辟内存

new操作符

基本用法

int* func()
{
    
	int* a = new int(10);
	return a;
}

int main() {
    

	int *p = func();

	cout << *p << endl;
	cout << *p << endl;

	//利用delete释放堆区数据
	delete p;

	//cout << *p << endl; //报错,释放的空间不可访问

	system("pause");

	return 0;
}

开辟数组

//堆区开辟数组
int main() {
    

	int* arr = new int[10];

	for (int i = 0; i < 10; i++)
	{
    
		arr[i] = i + 100;
	}

	for (int i = 0; i < 10; i++)
	{
    
		cout << arr[i] << endl;
	}
	//释放数组 delete 后加 []
	delete[] arr;

	system("pause");

	return 0;
}

引用

作用是给变量起一个别名(两个名字对应同一个变量),本质上是一种封装的指针。
数据类型 &别名 = 原名

int main() {
    

	int a = 10;
	int &b = a;

	cout << "a = " << a << endl;
	cout << "b = " << b << endl;

	b = 100;

	cout << "a = " << a << endl;
	cout << "b = " << b << endl;

	system("pause");

	return 0;
}

引用注意事项

必须初始化
不能更改

int main() {
    

	int a = 10;
	int b = 20;
	//int &c; //错误,引用必须初始化
	int &c = a; //一旦初始化后,就不可以更改
	c = b; //这是赋值操作,不是更改引用

	这里如果写成 &c=b也是不能更改引用的,因为&c是c的地址,等于是把c这个引用的地址赋值为了b

	cout << "a = " << a << endl;
	cout << "b = " << b << endl;
	cout << "c = " << c << endl;

	system("pause");

	return 0;
}

引用做函数参数

利用引用让形参成为实参的别名
相当于简化了地址传参

//1. 值传递
void mySwap01(int a, int b) {
    
	int temp = a;
	a = b;
	b = temp;
}

//2. 地址传递
void mySwap02(int* a, int* b) {
    
	int temp = *a;
	*a = *b;
	*b = temp;
}

//3. 引用传递
void mySwap03(int& a, int& b) {
    
	int temp = a;
	a = b;
	b = temp;
}

int main() {
    

	int a = 10;
	int b = 20;

	mySwap01(a, b);
	cout << "a:" << a << " b:" << b << endl;

	mySwap02(&a, &b);
	cout << "a:" << a << " b:" << b << endl;

	mySwap03(a, b);
	cout << "a:" << a << " b:" << b << endl;

	system("pause");

	return 0;
}

引用做函数返回值

作用:引用是可以作为函数的返回值存在的

注意:不要返回局部变量引用

用法:函数调用作为左值

//返回局部变量引用
int& test01() {
    
	int a = 10; //局部变量
	return a;
}

//返回静态变量引用
int& test02() {
    
	static int a = 20;
	return a;
}

int main() {
    

	//不能返回局部变量的引用
	int& ref = test01();
	cout << "ref = " << ref << endl;
	cout << "ref = " << ref << endl;

	//如果函数做左值,那么必须返回引用
	int& ref2 = test02();
	cout << "ref2 = " << ref2 << endl;
	cout << "ref2 = " << ref2 << endl;

	test02() = 1000;

	cout << "ref2 = " << ref2 << endl;
	cout << "ref2 = " << ref2 << endl;

	system("pause");

	return 0;
}

引用的本质

引用的本质在c++内部实现是一个指针常量(int * const,可以改变量值,不能改指针指向)

//发现是引用,转换为 int* const ref = &a;
void func(int& ref){
    
	ref = 100; // ref是引用,转换为*ref = 100
}
int main(){
    
	int a = 10;
    
    //自动转换为 int* const ref = &a; 指针常量是指针指向不可改,也说明为什么引用不可更改
	int& ref = a; 
	ref = 20; //内部发现ref是引用,自动帮我们转换为: *ref = 20;
    
	cout << "a:" << a << endl;
	cout << "ref:" << ref << endl;
    
	func(a);
	return 0;
}

常量引用

常量引用主要用来修饰形参,防止误操作

在函数形参列表中,可以加const修饰形参,防止形参改变实参

//引用使用的场景,通常用来修饰形参
void showValue(const int& v) {
    
	//v += 10;
	cout << v << endl;
}

int main() {
    

	//int& ref = 10;  引用本身需要一个合法的内存空间,因此这行错误
	//加入const就可以了,编译器优化代码,int temp = 10; const int& ref = temp;
	const int& ref = 10;

	//ref = 100;  //加入const后不可以修改变量
	cout << ref << endl;

	//函数中利用常量引用防止误操作修改实参
	int a = 10;
	showValue(a);

	system("pause");

	return 0;
}
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/HuiFeiDeTuoNiaoGZ/article/details/118049070

智能推荐

SAP ABAP 如何在程序中调用类的私有方法(CL_GUI_ALV_GRID)_sap 调用class_Gong JX的博客-程序员秘密

Please be aware that accessing private or protected data may have unpredictible consequences! Use it at your own risk.不介意使用,会发生不可预见的错误,需自行判断。类的private and protected方法只能在类内部使用,继承父类或者是在外部调用都是不可以的。但是有些情况要用到私有方法。CL_GUI_ALV_GRID可以通过接口IF_ALV_RM_GRID_FRI...

[assembly: AssemblyVersion("1.0.1.*")] 指定版本字符串不符合所需格式 - major[.minor[.build[.revision]]]..._anke7849631的博客-程序员秘密

报如下错误,解决方法:打开项目文件,修改打开项目文件修改:&lt;Deterministic&gt;true&lt;/Deterministic&gt; 为:&lt;Deterministic&gt;false&lt;/Deterministic&gt;转载于:https://www.cnblogs.com/ChangTan/p/10485669.h...

poj 1177 经典线段树+扫描线就周长并_suressay的博客-程序员秘密

可以参考陈宏的论文,就是将矩形的线段按x轴排序后每次利用前后直接线段覆盖y值的差值计算竖线的长度,利用前后两直线直接的x周距离乘不相交线段的条数计算横线的长度。具体可以看代码。本题坐标范围较小,可以不用离散化。#include#include#include#include#includeusing namespace std;const int nMax=5005;cons

UCI数据集数据的分析 ——葡萄酒数据_葡萄酒数据集分析_X_dmword的博客-程序员秘密

1、前言 葡萄酒是一种成分复杂的酒精饮料,不同产地、年份和品种的葡萄酒成分不同,这也是导致质量差异过大的重要因素。至今,质量评价主要还是依靠专家的感官。味道是最难理解的一种感官,因此用味蕾评价葡萄酒也就成为一件艰巨的任务。为了评估葡萄酒的质量,我们提出的方法就是根据酒的物理化学性质与质量的关系,找出高品质的葡萄酒具体与什么性质密切相关,这些性质又是如何影响葡萄酒的质量。2、数据准备...

net.sf.fmj.media.cdp.civil.CaptureDevicePlugger addCaptureDevices解决方法_vikione的博客-程序员秘密

<br />Spark运行时报错,解决办法,将spark源码目录E:/MyeclipseWorkspace/spark/build/lib/dist/windows 下的civil.dll 文件拷贝到C盘jdk的bin目录下,即可。    <br />  错误代码如下:<br />----错误代码 start----<br />010-11-15 10:56:11 net.sf.fmj.media.cdp.civil.CaptureDevicePlugger addCaptureDevices<br />警

解决Lombok不生效_zhangjie0523的博客-程序员秘密

解决lombok插件 不生效问题 启动项目报错

随便推点

Springboot 面试常问知识点、面试题合集_springboot的常问考点_柚子-youzi的博客-程序员秘密

这两年随着微服务的盛行,SpringBoot框架水到渠成的得到了高曝光,作为程序猿的我们,现在要是不知道一点SpringBoot相关的东西,貌似出去找工作都会被深深地鄙视。在这里我整理了一份常问Springboot 知识点和面试题,希望对大家有所帮助!由于篇幅有限,有需要完整版的朋友可以点一点下方链接免费领取!链接:1103806531暗号:CSDN1. SpringBoot 简介SpringBoot 是简化 Spring 应用开发的一个框架。他整合了 Spring 的技术栈,提供各种标准化的

Opencv之Homography_csdn330的博客-程序员秘密

     原文地址:Opencv日常之Homography什么是Homography 在图1中有两张书的平面图,两张图分别有四个相对位置相同的点,Homography就是一个变换(3*3矩阵),将一张图中的点映射到另一张图中对应的点 因为Homography是一个3*3矩阵,所以可以写成 H=⎡⎣⎢h00h10h20h01h11h21h02h12h22⎤⎦⎥H=[h00h01h02h10h11h1...

Java实现自定义敏感词库过滤_CxNull的博客-程序员秘密

最近接到一个需求,要添加一个敏感词管理模块,一如既往的CURD,敏感词我们添加到了自己的库里。然后进行一个自定义敏感词过滤,话不多说直接贴代码1、工具类这里只是最简单的得到敏感词进行转换,可以根据自己的业务需求进行填充package com.zylc.bixiang.business.keywords.web;import com.zylc.bixiang.business.keywords.domain.repository.BXKeyWordsMapper;import com.zy

Android自定义连接指定WiFi&热点开关_android自动连接指定热点_华~的博客-程序员秘密

Android自定义连接指定WiFi&amp;热点开关连接WiFi热点开关连接WiFi目的:可通过接受用户输入WiFi SSID、密码、密码类型控制设备连接至该WiFi步骤:开启WiFi、启动WiFi网络搜索、连接WiFi具体实现方式:(1)启用WIFI:检查设备目前是否已开启WiFi,如果未开启则开启WiFi,如果已开启并且处于已连接状态,则先断开当前连接,随后注册广播监听WiFi状...

异常org.hibernate.DuplicateMappingException:Table [xx] contains physical column name [xx]_小太阳DENGJIE的博客-程序员秘密

问题描述:SpringBoot+Hibernate项目中:实体类中属性上加入@Colunm注解,并指定注解的name为对应表字段时,报该错误。当将@Colunm注解去除或是在name属性中填入当前实体类名称,则不报错。详细错误信息如下:Caused by: org.hibernate.DuplicateMappingException: Table [jst_order] contains physical column name [is_cod] referred to by multiple

推荐文章

热门文章

相关标签