Java中参数传递问题

JAVA学习网 2017-12-18 00:00:03

前段时间对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对象内容不可变的原因?

因为目前还在基础阶段,学习的还不够深刻自己也有点疑问,以上理解可能存在偏差和错误,如有错误请大家指出。

 

阅读(769) 评论(0)