Java对象管理、赋值语句和值传递的一些理解
1.对象管理
参考来源:Java内存管理以及对象管理
基本概念
Java内存分为栈内存和堆内存。
栈(Stack):用来保存基本类型的变量和对象的引用变量
堆(Heap):使用来存放new创建的对象实例和数组
Hero hero=new Hero();
int const=1;
int[] i=new int[10];
const就是int类型的变量,hero是Hero类对象的引用变量。它们都是存放在栈中的。
new Hero(),new int[10]分别是创建的对象实例和数组,放在堆中。
栈中的变量可以取值new的数组或者对象的在堆中的首地址,这样就是通过栈中的引用变量来访问堆中的数组或者对象实例。换句话说,栈保存的是堆内存的访问地址。
对象的内存布局和访问定位
对象在内存中存储有三个区域:对象头,实例数据和对其填充。对象头:1.对象自身的运行数据,如哈希码;2.(类型指针)对象指向它的类型数据的指针,从而JVM可以确定这个对象是哪个类的实例。实例数据:对象真正存储的有效信息。对其填充:占位符,如补全为8字节的倍数。
对象的访问方式主要有:使用句柄和直接指针。
使用句柄 堆中会划分出一块内存用来作为句柄池,栈中的引用变量存储的是对象的句柄地址,而句柄中有对象实例数据和对象类型数据的地址。这两者分别在堆内存和方法区内存中(详解见参考来源)。
直接指针栈中的引用变量存储的是对象在堆中的存储地址。在此,就必须考虑如何放置访问类型数据的相关信息。
首先明确,java中的赋值是把右边所管理的对象交给左边管理,也就是让左右两边去管理同一个对象(把引用地址的值传递到了左边)。首先看一个例子:
2.赋值语句
参考来源:Java赋值的解释和辨析;java数组内存分配内存结构详解;java数组内存分配内存结构详解(源)
基本概念
赋值=:是引用地址值的传递。
对象实例情况:如下代码:
Hero hero1=new Hero("hero11");
Hero hero2;
Hero hero3=new Hero("hero33");
hero2=hero1;
//hero1=new Hero("d");
hero1=hero3;
hero3=new Hero("dd");
System.out.println(hero1);
System.out.println(hero3);
hero2=hero1,是把hero1的值赋给hero2,而这个值是实例化对象在堆中的存储地址。栈中的引用变量hero1和hero2通过引用地址去管理堆中的那个实例化对象(new Hero(“hero11”));hero3=new Hero(“dd”);hero3之前的值等于对象new Hero(“hero33”)的引用地址,现在其值被改变为new Hero(“dd”)的引用地址。所以hero33管理的对象也就发生了变化。
数组:如下代码
Hero[] heros1;
Hero[] heros2=new Hero[3];
for(int i=0;i<heros2.length;i++) {
heros2[i]=new Hero("hero "+i);
}
heros1=heros2;
System.out.println(heros1[0]);
heros1[0]=new Hero("hero x");
System.out.println(heros2[0]);
执行结果:
[hero 0]
[hero x]
heros1,heros2是存储在栈中的引用变量,其值是数组对象在堆中的存储地址。数组元素被存储在堆中,其值是也是一个引用地址的值,如下图所示:
students是数组对象的引用变量;lee和zhang是实例化对象的引用变量,students[0]和students[1]分别指向实例化对象zhang和lee。可以看出,数组元素在堆中的值等于实例化对象在堆中的引用地址。
在上面代码中,heros1=heros2;执行此代码之后,他们都指向了堆中同一个数组对象,而通过heros1[0]=new Hero(“hero x”);改变了堆中数组对象的一个数组元素,将这个数组元素的值改变为new Hero(“hero x”)在堆中的引用地址值,但是该数组元素在堆中的存储地址未发生改变,也就是说数组对象在堆中的内存地址未发生改变,所以对于heros2来讲,System.out.println(heros2[0]);输出之后就跟着变化为[hero x]。
2.值传递和引用传递
基本概念
JAVA中本质上只存在值传递,不存在引用传递。引用传递本质上是引用地址值的复制!
值类型:通俗来讲就是八大基本类型;在赋值时,直接在栈中生成值的类型
引用类型:是除了值类型以外的类型,包括类,接口,数组等;在初始化时将引用生成栈上,而值生成在堆上的这些数据类型
值传递指的是方法传参时,传递的是原内容的副本,因此对副本进行如何修改都不会影响原内容。
引用传递指的是方法传参时,传递的是参数本身,因此对参数进行任意修改都会影响原内容。这是“伪引用传递”,其实传递的是引用地址值的副本。引用传递感觉是针对引用类型来讲的,因引用类型变量存储的是地址值。
具体分析可以查看上面的参考来源!