前段时间对Java中参数传递问题有点困惑,不了解其中的含义。查阅了很多资料,这里谈谈自己对该问题的理解。
参数传递一般有两种,一种是“传值”,另一种是“传地址值”。传值是指在调用方法时,把参数的值传递给方法,而传地址值则是给方法提供参数的地址值。Java中的参数传递方法都为传值调用。下面我通过例子来验证。
1.基本类型的参数传递
1 public class ParamTransfer {
2 public int money;
3
4 public void amethod(int i) {
5 System.out.println("方法得到的i的值:" + i);
6 i = i * 5;
7 System.out.println("方法执行语句i=i*5后i的值:" + i);
8 System.out.println("money的值:" + this.money);
9 }
10
11 public static void main(String[] args) {
12 ParamTransfer pt = new ParamTransfer();
13 pt.money = 100;
14 pt.amethod(pt.money);
15 }
16
17 }
运行结果如下:
方法得到的i的值:100 方法执行语句i=i*5后i的值:500 money的值:100
注意对比形参i和money值的变化。这里我们把pt.money作为参数传递给amethod方法,该方法中i得到的值是100,这个我们从运行结果中可知。执行完i=i*5语句后我们发现当前对象的money属性值并没有发生改变,这就说明这里只是传了一个100数值给形参i,相当于把pt.money的值拷贝给i,i的变化并不会影响pt.money的值,仅仅是传数值。
2.引用类型的参数传递
1 class Time {
2 public int hour;
3 public int minute;
4 public int second;
5 }
6
7 public class ObjectParamTransfer {
8 Time time;
9
10 public static void main(String[] args) {
11 ObjectParamTransfer opt = new ObjectParamTransfer();
12 opt.time = new Time();
13 opt.time.hour = 11;
14 opt.time.minute = 22;
15 opt.time.second = 33;
16
17 System.out.println("time的属性值:");
18 System.out.println("hour=" + opt.time.hour);
19 System.out.println("minute=" + opt.time.minute);
20 System.out.println("second=" + opt.time.second);
21
22 opt.method(opt.time);//将对象引用作为参数传递给方法method
23 //对比执行方法后的变化
24 System.out.println("执行方法后的time的属性值");
25 System.out.println("hour=" + opt.time.hour);
26 System.out.println("minute=" + opt.time.minute);
27 System.out.println("second=" + opt.time.second);
28
29 }
30
31 private void method(Time t) {
32 System.out.println("参数t的属性值:");
33 System.out.println("hour=" + t.hour);
34 System.out.println("minute=" + t.minute);
35 System.out.println("second=" + t.second);
36 System.out.println("对t和time进行==比较,结果为:" + (t == this.time));
37
38 System.out.println("改变t的实例变量值");
39 t.hour = 44;
40 t.minute = 55;
41 t.second = 60;
42 }
43 }
运行结果如下:
time的属性值:
hour=11
minute=22
second=33
参数t的属性值:
hour=11
minute=22
second=33
对t和time进行==比较,结果为:true
改变t的实例变量值
执行方法后的time的属性值:
hour=44
minute=55
second=60
将对象引用opt.time传递给method方法后先输出参数t的属性值,发现和原来对象引用的属性值是相同的。进一步返回t==this.time的结果为true。改变t的实例变量值后,当前引用opt.time的属性值也随之发生改变。这就充分证明了参数t和对象引用opt.time指向同一对象,即它们指向的内存空间地址一致。那么问题来了,到底是opt.time这个引用传递了内存空间地址值给参数t还是它自己就是这个参数t呢?这我们不得而知。下面再通过一个例子来证明。
1 public class ObjectParamTransfer2 {
2 Time time1;
3 Time time2;
4
5 public static void main(String[] args) {
6 ObjectParamTransfer2 opt = new ObjectParamTransfer2();
7 opt.time1 = new Time();
8 opt.time2 = new Time();
9 opt.time1.hour = 12;
10 opt.time2.hour = 23;
11 System.out.println("交换前的值:");
12 System.out.println("time1.hour=" + opt.time1.hour);
13 System.out.println("time2.hour=" + opt.time2.hour);
14
15 opt.swap(opt.time1, opt.time2);
16 System.out.println("交换后的值:");
17 System.out.println("time1.hour=" + opt.time1.hour);
18 System.out.println("time2.hour=" + opt.time2.hour);
19 }
20
21 private void swap(Time t1, Time t2) {
22 Time temp;
23 temp = t1;
24 t1 = t2;
25 t2 = temp;
26 }
27
28 }
运行结果如下:
交换前的值:
time1.hour=12
time2.hour=23
交换后的值:
time1.hour=12
time2.hour=23
这里写了一个交换两个引用的swap的方法。但是我们从结果中发现调用该方法后输出结果并没有发生改变。说明该方法并没有对opt.time1和opt.time2这两个引用进行交换。在该方法中只是对参数t1,t2进行交换,验证了上个例子中引用传递的是内存空间地址值,并非参数就是该引用(即传的不是引用)。对参数的操作并不会影响原来的引用,只会影响参数和引用所指的同一个内存空间里面的内容。这就证明了引用类型参数传递并不是传引用,而只是传该引用的内存地址值。
3.参数传递中的一种特殊情况
在上面的例子中我们已经知道了基本类型的参数传递传的是传数值,引用类型的参数传递传的是内存地址值。但我在学习过程中碰到一种特殊情况:
1 public class Test {
2 public static void main(String[] args) {
3 String str = new String("abc");
4 change1(str);
5 System.out.println(str);
6 }
7
8 private static void change1(String str1) {
9 str1 += "123";
10 System.out.println(str1);
11 }
12 }
运行结果:
abc123
abc
按上面得出的结论来看,引用类型的参数传递传递的是内存地址值,str和str1指向的是同一个内存空间,str1将内存空间里面的内容改变后输出str也应该发生改变。但是str指向的内存空间里的内容“abc”并没有发生变化。这里我理解为String类型的引用传递传的是内容,并不是内存空间地址值。(与第一个例子中传数值类似)并不会影响原来引用所指的内存空间的内容。可能这也是String对象内容不可变的原因?
因为目前还在基础阶段,学习的还不够深刻自己也有点疑问,以上理解可能存在偏差和错误,如有错误请大家指出。