1、引用传递
引用传递也称为传地址,指的是在方法调用时,传递的参数是按引用进行传递,其实传递的是引用的地址,也就是变量所对应的内存空间的地址。
方法调用时,实际参数的引用——地址被传递给方法中相对应的形式参数,即形式参数和实际参数拥有相同的存储单元。在方法执行过程中,对形式参数的操作实际上就是对实际参数的操作,因此形式参数值的改变将会影响实际参数的值。
2、堆内存与栈内存
(1)堆内存:堆内存可以理解为一个对象的具体信息,每一个对象保存的只是属性信息,每一块堆内存的开辟都要通过关键字new来完成。
(2)栈内存:可以理解为一个整型变量(只能够保存一个数值),其中保存的是一块(只能保存一块)堆内存空间的内存地址数值,但是为了方便理解,现在可以假设其保存的是对象的名字。
class Book {
String title;
double price;
public void printInfo() {
System.out.println("title:" + this.title);
System.out.println("price:" + this.price);
}
}
public class Reference {
public static void main(String[] args) {
Book book = null; // 声明对象
book = new Book(); // 实例化一个对象
book.title = "Java程序设计";
book.price = 39.8;
book.printInfo(); // 通过对象调用,不是直接调用
}
}
【结果】
下面看一下该程序的内部执行过程:
如果把“book = new Book() ;”注释掉,此时会出现如下的错误提示:
“NullPointerException”表示空指向异常,指的是使用了一个未实例化的对象(未开辟堆内存空间的对象)进行了类中属性或方法的调用的时候就会出现本异常信息。 所以对象使用之前一定要开辟堆内存空间。
在程序中一个关键字new产生一个对象,就开辟一个新空间,如果有两个关键字new,就表示要开辟两个新内存空间,此处的两个对象都占有各自的内存空间,彼此的操作应该是独立的。也即,只要用了new,不管在何种情况下,都表示要开辟新的内存空间。
3、引用数据类型的传递
在Java中,类本身就是引用数据类型,而对于引用数据类型实际上就相当于其他语言之中的指针概念。
在Java中对于方法参数的传递,对象是传递引用,基本数据类型是传递值。下面看一个传递基本数据类型的例子:
public class ReferenceDemo2 {
public static void main(String[] args) {
int i =3,
j = 4;
swap(i, j);
System.out.println("i = " + i);
System.out.println("j = " + j);
}
public static void swap(int i, int j) {
int temp = i;
i = j;
j = temp;
}
}
【结果】
从本范例中可以看到,引用数据类型的传递并没有改变数据本身的值。因为参数中传递的是基本类型 i 和 j 的备份,在函数中交换的也是那份备份的值而不是数据本身。
再看下面这个例子:引用传递
public class ReferenceDemo2 {
public static void main(String[] args) {
int[] count = {1,2,3,4,5};
System.out.println("方法执行前 : count[0] = " + count[0]);
swap(count);
System.out.println("方法执行后 : count[0] = " + count[0]);
}
public static void swap(int[] count) {
count[0] = 0;
System.out.println("在方内部 : count[0] = " + count[0]);
}
}
【结果】
从本范例中可以看到,在方法中传递引用数据类型int数组,实际上传递的是其引用count的备份,它们都指向数组对象,在方法中可以改变数组对象的内容。即:对复制的引用所调用的方法更改的是同一个对象。
再来看一个关于对象引用传递的例子:
class Person {
String name;
int age;
}
public class ReferenceDemo3 {
public static void main(String[] args) {
Person p1 = null; // 声明p1,此对象值为null,尚未实例化
Person p2 = null;
p1 = new Person(); // 实例化对象p1
p1.age = 20;
p1.name = "小光";
p2 = p1; // 将p1的引用赋给p2
System.out.println("姓名 : " + p2.name);
System.out.println("年龄 : " + p2.age);
p1 = null;
}
}
【结果】
下而是具体的引用过程:
所谓的引用传递,指的是一块堆内存空间,同时被多个栈内存所指向。引用传递的核心认识:不同的栈内存如果指向了同一块堆内存之中,所做的修改将影响所有的栈内存。
可以发现,每一块栈内存只能够保存一块堆内存的地址,但是反过来,一块堆内存可以同时被多个栈内存所指向,在这种情况下,如果要改变某一个栈内存的保存地址内容,则必须先断开已有的堆内存地址连接,才可以指向新的堆内存空间,而如果一块堆内存空间没有任何的栈内存所指向的话,那么这块空间就将成为垃圾,所有的垃圾将等待被JVM中的GC(Garbage Collector)进行不定期的收集,同时进行内存空间的释放。