咱们以前一直在使用“对象”这个概念,但没有探讨对象在内存中的具体存储方式。这方面的讨论将引出“对象引用”(object reference)这一重要概念。
html
咱们沿用以前定义的Human类,并有一个Test类:app
= Human(160 Human(.height = growHeight(.height = .height +
外部能够调用类来建立对象,好比上面在Test类中:ide
Human aPerson = new Human(160);
建立了一个Human类的对象aPerson。 焦做国医胃肠医院怎么坐车:http://jz.lieju.com/zhuankeyiyuan/37325304.htm函数
上面是一个很是简单的表述,但咱们有许多细节须要深刻:post
首先看等号的右侧。 new是在内存中为对象开辟空间。具体来讲,new是在内存的 堆(heap)上为对象开辟空间。这一空间中,保存有对象的数据和方法。测试
再看等号的左侧。aPerson指代一个Human对象,被称为 对象引用(reference)。实际上,aPerson并非对象自己,而是相似于一个指向对象的指针。aPerson存在于内存的 栈(stack)中。spa
当咱们用等号赋值时,是将右侧new在堆中建立对象的地址赋予给对象引用。设计
这里的内存,指的是JVM (Java Virtual Machine)虚拟出来的Java进程内存空间。内存的堆和栈概念可参考 Linux从程序到进程。指针
http://dalian.qd8.com.cn/yiyao/xinxi21_3709995.htmlcode
对象引用
栈的读取速度比堆快,但栈上存储的数据受到有效范围的限制。在C语言中,当一次函数调用结束时,相应的栈帧(stack frame)要删除,栈帧上存储的参量和自动变量就消失了。Java的栈也受到一样的限制,当一次方法调用结束,该方法存储在栈上的数据将清空。在 Java中, 全部的(普通)对象都储存在堆上。所以, new关键字的完整含义是, 在堆上建立对象。
基本类型(primitive type)的对象,好比int, double,保存在栈上。当咱们声明基本类型时,不须要new。一旦声明,Java将在栈上直接存储基本类型的数据。因此,基本类型的变量名表示的是数据自己,不是引用。
引用和对象的关系就像风筝和人。咱们看天空时(程序里写的),看到的是风筝(引用),但风筝下面对应的,是人(对象):
引用和对象分离;引用指向对象
尽管引用和对象是分离的,但咱们全部通往对象的访问 必须通过引用这个“大门”,好比以 引用.方法() 的方式访问对象的方法。在Java中,咱们不能跳过引用去直接接触对象。再好比,对象a的数据成员若是是一个普通对象b,a的数据成员保存的是指向对象b的引用 (若是是基本类型变量,那么a的数据成员保存的是基本类型变量自己了)。
在Java中,引用起到了指针的做用,但咱们不能直接修改指针的值,好比像C语言那样将指针值加1。咱们只能经过引用执行对对象的操做。这样的设计避免了许多指针可能引发的错误。
当咱们将一个引用赋值给另外一个引用时,咱们实际上复制的是对象的地址。两个引用将指向同一对象。好比 dummyPerson=aPerson;,将致使:
一个对象能够有多个引用 (一我的能够放多个风筝)。当程序经过某个引用修改对象时,经过其余引用也能够看到该修改。咱们能够用如下Test类来测试实际效果:
public class Test { public static void main(String[] args) { Human aPerson = new Human(160); Human dummyPerson = aPerson; System.out.println(dummyPerson.getHeight()); aPerson.growHeight(20); System.out.println(dummyPerson.getHeight()); } }
咱们对aPerson的修改将影响到dummyPerson。这两个引用实际上指向同一对象。
因此,将一个引用赋值给另外一个引用,并不能复制对象自己。咱们必须寻求其余的机制来复制对象。
随着方法调用的结束, 引用和 基本类型变量会被清空。因为对象存活于堆,因此对象所占据的内存不会随着方法调用的结束而清空。进程空间可能很快被不断建立的对象占满。Java内建有 垃圾回收(garbage collection)机制,用于清空再也不使用的对象,以回收内存空间。
垃圾回收的基本原则是,当存在引用指向某个对象时,那么该对象不会被回收; 当没有任何引用指向某个对象时,该对象被清空。它所占据的空间被回收。
上图假设了某个时刻JVM中的内存状态。Human Object有三个引用: 来自栈的aPerson和dummyPerson,以及另外一个对象的数据成员president。而Club Object没有引用。若是这个时候垃圾回收启动,那么Club Object将被清空,而Human Object来自Club Object的引用(president)也随之被删除。
垃圾回收是Java中重要的机制,它直接影响了Java的运行效率。我将在之后深刻其细节。
当咱们分离了引用和对象的概念后,Java方法的参数传递机制实际上很是清晰: Java的参数传递为 值传递。也就是说,当咱们传递一个参数时,方法将得到该参数的一个拷贝。
实际上,咱们传递的参数,一个是基本类型的变量,另外一个为对象的引用。
基本类型变量的值传递,意味着变量自己被复制,并传递给Java方法。Java方法对变量的修改 不会影响到原变量。
引用的值传递,意味着对象的地址被复制,并传递给Java方法。Java方法根据该引用的访问将 会影响对象。
在这里有另外一个值得一提的状况: 咱们在方法内部使用new建立对象,并将该对象的引用返回。若是该返回被一个引用接收,因为对象的引用不为0,对象依然存在,不会被垃圾回收。
new
引用,对象
被垃圾回收的条件
参数: 值传递