前言:
学习数据结构(java),基于数组的基于数组的顺序表、堆和队列相对来说比较简单,即使没有理解java的内存的分配,你会发现没有什么关系。但是当你学习链表的时候,会遇到很多的问题。java没有指针,那么java如何将一个节点指向另一个节点的,他们的实现的原理是什么。还有你会慢慢倒地对象是什么?什么是实例化?慢慢你又发现,那什么是引用呢?
为了解决上述的问题,质询老师,百度。我感觉有必要了解一下java的内存分配的机制和java的引用。
在Java代码中,常常会使用到这样的类的声明实例化:
Person per = new Person();
//这其实是包含了两个步骤,声明和实例化
Person per = null; //声明一个名为Person类的对象per
per = new Person(); // 实例化这个per对象
声明 指的是创建类的对象的过程;
实例化 指的是用关键词new来开辟内存空间。
它们在内存中的划分是这样的:
要好好的理解这张图,通过new已经将per(引用)指向堆内存。
那什么是栈内存(heap)和栈内存(heap)呢?
栈内存:
在函数中定义的一些基本类型的变量和对象的引用变量都在函数的栈内存中分配。栈内存主要存放的是基本类型类型的数据 如( int, short, long, byte, float, double, boolean, char) 和对象句柄。注意:并没有String基本类型、在栈内存的数据的大小及生存周期是必须确定的、其优点是寄存速度快、栈数据可以共享、缺点是数据固定、不够灵活。
栈的共享:
1
2
3
4
5
6
7
8
9
10
11
12
13 1String str1 = "myString";
2
3String str2 = "myString";
4
5System.out.println(str1 ==str2 );
6
7//注意:这里使用的是str1 ==str2,而不是str1.equals(str2)的方式。
8
9//因为根据JDK的说明,==号只有在两个引用都指向了同一个对象时才返回真值
10
11//而str1.equals(str2),只是比较两个字符串是否相等
12
13
结果为True,这就说明了str1和str2其实指向的是同一个值。
上述代码的原理是,首先在栈中创建一个变量为str1的引用,然后查找栈中是否有myString这个值,如果没找到,就将myString存放进来,然后将str1指向myString。接着处理String str2 = “myString”;;在创建完str2 的引用变量后,因为在栈中已经有myString这个值,便将str2 直接指向myString。这样,就出现了str1与str2 同时指向myString。
特别注意的是,这种字面值的引用与类对象的引用不同。假定两个类对象的引用同时指向一个对象,如果一个对象引用变量修改了这个对象的内部状态,那么另一个对象引用变量也即刻反映出这个变化。相反,通过字面值的引用来修改其值,不会导致另一个指向此字面值的引用的值也跟着改变的情况。如上例,我们定义完str1与str2 的值后,再令str1=yourString;那么,str2不会等于yourString,还是等于myString。在编译器内部,遇到str1=yourString;时,它就会重新搜索栈中是否有yourString的字面值,如果没有,重新开辟地址存放yourString的值;如果已经有了,则直接将str1指向这个地址。因此str1值的改变不会影响到str2的值。
堆内存:
堆内存用来存放所有new 创建的对象和 数组的数据
1
2
3
4
5
6
7
8
9
10
11
12
13 1String str1 = new String ("myString");
2
3String str2 = "myString";
4
5System.out.println(str1 ==str2 ); //False
6
7String str1 = new String ("myString");
8
9String str2 = new String ("myString");
10
11System.out.println(a==b); //False
12
13
创建了两个引用,创建了两个对象。两个引用分别指向不同的两个对象。以上两段代码说明,只要是用new()来新建对象的,都会在堆中创建,而且其字符串是单独存值的,即使与栈中的数据相同,也不会与栈中的数据共享。