Java开发——6.内存分配(堆、栈以及参数的值传递+引用传递)

简介: 内存分配(堆、栈以及参数的值传递+引用传递)

内存:

内存是计算机中的重要原件,临时存储区域,作用是运行程序。

我们编写的程序是存放在硬盘中的,在硬盘中的程序是不会运行的。

必须放进内存中才能运行,运行完毕后会清空内存

Java虚拟机要运行程序,必须要对内存进行空间的分配和管理。


此处补充内存和硬盘的区别:

1.内存是指安装在系统板上的随机存取内存,而硬盘驱动器是磁盘的主轴,也称为硬盘。

2.内存容量范围从 128 MB 到 4 GB,而硬盘容量范围从 320 GB 到 1TB。

3.内存中的存储类型是临时的(如果一断电内存中的数据就会丢失),而硬盘中的存储类型是永远。

4.内存:不允许我们存储个人数据;而是用于存储计算机数据。它读取和写入数据,并且会继续保存在内存,除非系统关闭。硬盘是一种可预测的存储器,可以让用户存储和擦除数据;存储在硬盘上的所有数据在使用期间和使用后都不会自动清除。

拓展:

RAM,一般指随机存取存储器。

随机存取存储器(英语:Random Access Memory,缩写:RAM),也叫主存,是与CPU直接交换数据的内部存储器。RAM即大家常说的运行内存,简称运存,是一种在手机中用来暂时保存数据的元件,相当于电脑中的内存条。


Java中的内存分配

image.png


publicclassDemo{
publicstaticvoidmain(String []args){
intnum=5;
intarr [] =newint []{1,2,3};
    }
}


栈内存:一般存放基本数据类型的值;当在一段代码块中定义一个变量时,java就在栈中为这个变量分配内存空间,当超过变量的作用域后,java会自动释放掉为该变量分配的内存空间,该内存空间可以立刻被另作他用。


堆内存:一般存放引用数据类型的值;在栈中存储的变量就变成了数组或者对象的引用变量,以后就可以在程序中使用栈内存中的引用变量来访问堆中的数组或者对象,引用变量相当于为数组或者对象起的一个别名,或者代号。

引用变量是普通变量,定义时在栈中分配内存,引用变量在程序运行到作用域外释放。

例如:

int arr [] = new int []{1,2,3};

int sum = arr[2]; sum就是一个变量。

而数组和对象本身仍存放在堆内存中,数组和对象在没有引用变量指向它的时候,才变成垃圾,不能再被使用,但是仍然占着内存,在一个不确定的时间被垃圾回收器释放掉才能彻底释放内存。

这个也是Java比较占内存的主要原因,实际上,栈中的变量指向堆内存中的变量,这就是 Java 中的指针!

image.png



值传递的本质:

在参数传递的过程中,原来参数值不会发生改变!



publicstaticvoidmain(String[] args) {
//num 就是实参:实参是定义在方法之外类之中intnum=15;
System.out.println("值传递之前:"+num);
zhiChuanDi(num);
System.out.println("值传递之后:"+num);
}
publicstaticvoidzhiChuanDi(intnumber){
//此处的number 就是形参:形参是定义在方法之中的number=150;
    }

图解分析:

主要原因:方法存储在栈内存中的,栈内存中的方法会随着方法的调用完成而消失;所以尽管num被当做参数传进了zhiChuanDi()的有参方法中,并且发生了重赋值的步骤,但是整个过程并不会影响原来的num实参。相当于复制了一份num,并把复制后的num当做形参传递给了zhiChuanDi()的有参方法,实际的num是不变的。


image.png


引用传递的本质:

在进行参数传递的过程中,发生了参数值的改变。


publicstaticvoidmain(String[] args) {
intarr[] =newint[]{12,18,20};
System.out.println("引用传递之前:"+arr[0]);
yinYongChuanDi(arr);
System.out.println("引用传递之后:"+arr[0]);
Stringstr="小红";
System.out.println("引用传递之前:"+str);
yinYongChuanDi2(str);
System.out.println("引用传递之后:"+str);
    }
publicstaticvoidyinYongChuanDi(int []arr){
arr[0] =66;
    }
publicstaticvoidyinYongChuanDi2(Stringstr){
str="小明";
    }

图解分析:

这个过程相当于,人的名字一样;大多数人的名字都会重复,但是每个人对应的身份证是不同的!因为new对象,对象的值是存在于堆内存中的,要想获取堆内存中的具体指,是通过底层的地址值进行访问的,所以以参数传过去的arr数组,实际上是传的一个具体实际的arr数组,所以对arr[0]进行参数重赋值的过程,是对底层堆内存中的数据进行修改的;所以数据在进行参数传递的过程中发生了改变。

image.png

但是:我下面举了一个例子,是String类型的字符串,在进行参数的传递过程中,值并没有发生改变。虽然String类型也是引用数据类型,但是他发生参数传递的过程相当于基本数据类型,因为他并没有new一个对象!所以此处的String为特殊例子。

相关文章
|
6天前
|
Java 索引
java中的栈(利用数组实现栈)
这篇文章通过Java代码示例介绍了如何使用数组实现栈操作,包括栈的初始化、入栈、出栈、判断栈满和空以及遍历栈的方法。
java中的栈(利用数组实现栈)
|
5天前
|
存储 Java 开发者
Java 内存模型与垃圾回收机制的深度剖析
在Java的世界里,理解其内存模型和垃圾回收机制是提升编程效率的关键。本文将通过浅显易懂的语言,带你深入了解Java的内存分配原理和垃圾回收过程,让你对Java的性能调优有更深刻的认识。
|
6天前
|
Swift iOS开发
iOS开发-属性的内存管理
【8月更文挑战第12天】在iOS开发中,属性的内存管理至关重要,直接影响应用性能与稳定性。主要策略包括:`strong`(强引用),不维持对象生命期,可用于解除循环引用;`assign`(赋值),适用于基本数据类型及非指针对象属性;`copy`,复制对象而非引用,确保对象不变性。iOS采用引用计数管理内存,ARC(自动引用计数)自动处理引用增减,简化开发。为避免循环引用,可利用弱引用或Swift中的`[weak self]`。最佳实践包括:选择恰当的内存管理策略、减少不必要的强引用、及时释放不再使用的对象、注意block内存管理,并使用Xcode工具进行内存分析。
|
8天前
|
安全 Java 程序员
深入浅出Java内存模型:探索JMM的奥秘
在Java编程世界中,理解其内存模型(JMM)是提升代码性能和确保线程安全的关键。本文将带你走进Java内存模型的大门,通过浅显易懂的方式揭示其工作原理,并指导你如何在实际开发中有效利用JMM来避免常见的并发问题。
|
4天前
|
存储 监控 算法
掌握Java内存管理:从入门到精通
在Java的世界里,内存管理是程序运行的心脏。本文将带你走进Java内存管理的奥秘,从基础概念到高级技巧,一步步揭示如何优化你的Java应用。准备好迎接挑战,让我们共同揭开高效内存使用的面纱!
|
6天前
|
存储 缓存 安全
Java内存模型详解
该文章主要介绍了Java内存模型的相关概念和技术细节,包括Java内存模型的定义、缓存一致性策略、内存交互操作、内存屏障等。
|
14天前
|
Java
java是值传递还是引用传递
本文澄清了Java中参数传递的常见误解,总结出Java采用“值传递”的方式。对于基本类型,传递其值的拷贝,方法内修改不影响原值;而对于对象类型,则传递其引用地址的拷贝,尽管是拷贝,但因指向同一对象,故方法内的修改会影响原对象状态。形参仅在方法内部有效,而实参则是调用方法时传递的具体值。通过示例和比喻(如复刻仓库钥匙),形象地解释了值传递、引用传递及Java特有的“共享对象传递”概念,帮助理解不同情况下参数传递的行为差异。
|
15天前
|
Java
java中的值传递和引用传递
【8月更文挑战第3天】在Java中,值传递用于基本数据类型,传递的是值的副本,因此方法内的修改不影响原值;而引用传递用于对象和数组,虽传递的是引用的副本,但对对象内容的修改会影响原始对象。理解这两者对于正确处理方法调用及参数至关重要。
|
17天前
|
Java
java中的值传递和引用传递
【8月更文挑战第2天】在Java中,基本数据类型如`int`、`double`等采用值传递,传递的是变量值的副本,因此方法内的修改不影响原变量。对象类型则通过引用传递,传递的是对象引用的副本,允许方法内修改原对象。例如,对`StringBuilder`对象的修改会影响原始对象。
|
2月前
|
Java
Java的值传递与“引用传递”辨析
Java的值传递与“引用传递”辨析
14 0