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(); // 通过对象调用,不是直接调用
	}
}

【结果】

java 引用目录_java 引用目录


下面看一下该程序的内部执行过程:

java 引用目录_引用传递_02


如果把“book = new Book() ;”注释掉,此时会出现如下的错误提示:

java 引用目录_内存空间_03

“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;
	}
}

【结果】

java 引用目录_内存空间_04


从本范例中可以看到,引用数据类型的传递并没有改变数据本身的值。因为参数中传递的是基本类型 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]);
	}
}

【结果】

java 引用目录_java 引用目录_05


从本范例中可以看到,在方法中传递引用数据类型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;
	}
}

【结果】

java 引用目录_引用传递_06


下而是具体的引用过程:

java 引用目录_Java_07

所谓的引用传递,指的是一块堆内存空间,同时被多个栈内存所指向。引用传递的核心认识:不同的栈内存如果指向了同一块堆内存之中,所做的修改将影响所有的栈内存。

可以发现,每一块栈内存只能够保存一块堆内存的地址,但是反过来,一块堆内存可以同时被多个栈内存所指向,在这种情况下,如果要改变某一个栈内存的保存地址内容,则必须先断开已有的堆内存地址连接,才可以指向新的堆内存空间,而如果一块堆内存空间没有任何的栈内存所指向的话,那么这块空间就将成为垃圾,所有的垃圾将等待被JVM中的GC(Garbage Collector)进行不定期的收集,同时进行内存空间的释放。