Java:传值还是传引用?

这是一个Java的经典问题,大部分人从C,C++语言入门,C语言有三种传递方式:值传递,地址传递和引用传递。详细的对C语言指针,引用的我个人的理解,见链接

Java所有操作都是传值操作!都是传值操作!都是传值操作!重要的事情说三遍。

疑问?那为什么别人讲的时候都是说,java的基本数据类型都是传值,所有的自定义数据(类的对象)都是传引用??

很简单,因为这样好理解,意思是说:“同学们,如果我们把一个基本数据类型的值(变量)传递给一个函数的形参,那么无论我们对这个变量怎么操作,函数运行完之后,并不会改变这个变量的值!(到这里都是对的)但是如果我们把一个类的实例(对象)作为参数传递给函数,那么我们在函数里面对这个对象的改变,会实际地改变这个对象的值!(这里就不完全正确了)”。

对于Java的对象与引用的理解,请参见:浅谈Java中的对象和引用。我这里简单说一下Person person = new Person("张三"),这里面new Person("张三")是类Person的实例(或者说是对象),person是这个实例(对象)的引用。

Java的参数传递都是值传递!!!

先贴一篇博文:理解Java中的引用传递和值传递,如果你觉得这个人讲的很有道理,那么你是麻瓜(哇!马老师附体!)。我先不解释,去看下这篇博文的1楼评论!

我们先定义,什么是值传递,什么是引用传递。

值传递:方法调用时,实际参数把它的值的副本传递给对应的形式参数。特点:此时内存中存在两个相等的基本类型,即实际参数和形式参数后面方法中的操作都是对形参这个值的修改,不影响实际参数的值

引用传递:方法调用时,实际参数的引用(地址,而不是参数的值)被传递给方法中相对应的形式参数,函数接收的是原始值的内存地址;特点:在方法执行中,形参和实参内容相同,指向同一块内存地址,方法执行中对引用的操作将会影响到实际对象

值传递实在是太简单了,大家应该都容易理解,既然传递的是实际参数的副本,那么更改这个副本,跟原来的变量没有一点关系。

但是对于引用传递,一般的支持引用的例子我也不说了。看一个稍微特殊的例子:如果我们把下面的例子理解为引用传递,输出应该是: LI Si 才对!因为zhangsan这个(new Person("ZHANG San")的)引用指向了新的对象!但实际结果是zhangsan这个引用指向的对象的名字并没有变!

 1 public class ValueOrRef {
 2     public static void main(String[] args) {
 3         Person zhangsan = new Person("ZHANG San");
 4         changePerson(zhangsan);
 5         zhangsan.printName();
 6     }
 7     public static void changePerson(Person person){
 8         person = new Person("LI Si");
 9     }
10 }
11 
12 class Person {
13     String name = "default";
14     public Person(String name) {
15         this.name = name;
16     }
17     public void changeName(String name){
18         this.name = name;
19     }
20     void printName() {
21         System.out.println(this.name);
22     }
23 }

所以这里,我们把这种传递也理解为”值传递“只不过这里的值,是一个”引用”的值!也就是我们把实际参数(一个引用)拷贝一份赋值给形式参数,形式参数进行操作。当形式参数对本身的对象进行了变动操作,这里的效果跟引用传递是相同的。但是一旦给形式参数进行赋值类型的操作,这个赋值操作并不会像C语言的引用那样,把这个赋值的效果反映在函数运行结束之后!!

总结:不要纠结与值传递与引用传递的概念了,我们明白函数执行的过程就好了!为了更好地理解整个运行过程的变化,请参考:Java:按值传递还是按引用传递详细解说(其中有图片演示,很清楚)。

posted @ 2017-04-11 21:09  chen-kh  阅读(19854)  评论(3编辑  收藏  举报