值传递:(形式参数类型是基本数据类型):方法调用时,实际参数把它的值传递给对应的形式参数,形式参数只是用实际参数的值初始化自己的存储单元内容,是两个不同的存储单元,所以方法执行中形式参数值的改变不影响实际参数的值。
引用传递:(形式参数类型是引用数据类型参数):也称为传地址。方法调用时,实际参数是对象(或数组),这时实际参数与形式参数指向同一个地址,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,这个结果在方法结束后被保留了下来,所以方法执行中形式参数的改变将会影响实际参数。
//按值传递
public class shiyan { //先创建一个更改变量的函数,并加上了打印的函数用于比较 public void change(int c){ c=c*5; System.out.println(c); } public static void main(String[] args) { // TODO Auto-generated method stub int p=5; //先声明一个变量p=5 shiyan c=new shiyan(); //创建对象c //利用c的成员函数改变p值,使p=p*5 c.change(p); //输出的是理想值25 System.out.print(p); //奇怪输出的是5 } }
输出结果:25 5
分析:这就是按值传递的效果,因为此时传入的不是p本身,只是p的值,传递的是p本身的是副本。
所以打印出的是(p副本的值)的改变结果25,而后一个打印出的是p的真身没有改变过因此是5
//按引用传递
package yinyong; //先定义一个学生类 public class Student { //设计重载构造函数,为了后面的演示方便 public Student(){ } //用于取名的构造函数 public Student(String p){ name=p; } //用于改名的的函数,不过得改成别人的名字 public void changName(Student p){ this.name=p.name; } public String name; }
package yinyong; public class teacher { public static void main(String[] args) { // TODO Auto-generated method stub //创建a这个人给a学生取名李明 Student a=new Student("李明"); //创建b这个人给b学生取名赵明 Student b=new Student("赵明"); //创建c这个人,还未起名 Student c=new Student(); //分别输出a,b两个人的名字 System.out.println("a同学名字是"+a.name); System.out.println("b同学名字是"+b.name); //将c人的名字指向b,实质是将b的地址赋给c,这样c,b同指一片内存区域 c=b; System.out.println("c同学名字是"+c.name); //果然c的名字变为了赵明 b.changName(a); //利用改名函数改变b的名字,将a对象传入将a的名字赋给b System.out.println("b同学名字是"+b.name); //结果是a的名字李明 } }
输出:a同学名字是李明
b同学名字是赵明
c同学名字是赵明
b同学名字是李明
分析: 由于是按引用调用所以,传递的时候时刻都是把对象的真身传递过去,并且可以改变真身的属性和方法。传递时时候也把真身的地址传了过去,这样引用时就指向了同一片地址内存空间。