Java 为值传递而不是引用传递

——reference Java is Pass by Value and Not Pass by Reference

其实这个问题是一个非常初级的问题,相关的概念初学者早已掌握,但是时间长了还是容易混淆,特此总结一下

一、值传递和引用传递

  首先这里我们先看下两者的异同:

  • 值传递:方法调用时,实际参数把它的值传递给对应的形式参数,方法执行中形式参数值的改变不影响实际参 数的值。
  • 引用传递:也称为传地址。方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,在方法执行中,对形式参数的操作实际上就是对实际参数的操作,方法执行中形式参数值的改变将会影响实际参数的值。

二、Java总是值传递而非引用传递

举例:

Balloon.java

 1 package com.journaldev.test;
 2  
 3 public class Balloon {
 4  
 5     private String color;
 6  
 7     public Balloon(){}
 8      
 9     public Balloon(String c){
10         this.color=c;
11     }
12      
13     public String getColor() {
14         return color;
15     }
16  
17     public void setColor(String color) {
18         this.color = color;
19     }
20 }

之后我们做一个简单测试,交换两个对象

Test.java

 1 package com.journaldev.test;
 2  
 3 public class Test {
 4  
 5     public static void main(String[] args) {
 6  
 7         Balloon red = new Balloon("Red"); //memory reference 50
 8         Balloon blue = new Balloon("Blue"); //memory reference 100
 9          
10         swap(red, blue);
11         System.out.println("red color="+red.getColor());
12         System.out.println("blue color="+blue.getColor());
13          
14         foo(blue);
15         System.out.println("blue color="+blue.getColor());
16          
17     }
18  
19     private static void foo(Balloon balloon) { //baloon=100
20         balloon.setColor("Red"); //baloon=100
21         balloon = new Balloon("Green"); //baloon=200
22         balloon.setColor("Blue"); //baloon = 200
23     }
24  
25     //Generic swap method
26     public static void swap(Object o1, Object o2){
27         Object temp = o1;
28         o1=o2;
29         o2=temp;
30     }
31 }

输出如下:

1 red color=Red
2 blue color=Blue
3 blue color=Red

 从上面的代码可以得出:

  • 两个对象在被调函数中交换并不能影响主调函数中的两个对象
  • 但是在被调函数中对对象进行修改(对象的某一域),那么在主调函数中是可以反映出来的

第一点我们就可以看出Java确实是值传递的,然而第二点又让我们出现疑惑,为何对象的修改会影响到主调函数

这涉及到Java中的数据存储方式,堆栈存储在(Java 堆内存与栈内存异同(Java Heap Memory vs Stack Memory Difference)

Java中除了基本类型(【Java心得总结一】Java基本类型和包装类型解析)外,其他类型声明都是在栈中保存一个指向堆中的引用,而对象的数据都是在堆中保存的,如图:

 

 

所以在函数调用时,如果传递对象,Java确实是值传递,开辟一个新的存储空间存放形参,并将实参的值传递过来,但是两者指向同一处堆的存储空间,所以造成了上述交换无用,但是改变对象中的数据有用的结果

 

posted @ 2015-05-12 22:50  xlturing  阅读(721)  评论(0编辑  收藏  举报