Java中方法之间的参数传递问题,一直是我之前比较疑惑的地方。有人说Java中只有值传递,没有引用传递;也有人说Java中参数传递如果是普通类型,那么就是值传递,如果是对象,那么就是引用传递。很多人对这个问题的理解不尽相同。下面我们就针对上面的一些观点,深入的讲解一下。
参考资料:《深入理解Java核心技术》
如果你也有如下几个观点,那么一定要好好看接下来的类容:
上结论:
Java 中方法参数传递方式是按值传递
- 如果参数是基本类型, 传递的是基本类型的字面量值的拷贝(传值调用)
- 如果传递的参数是引用类型,传递的是该参数所引用的对象在堆内存中的地址的拷贝。(传共享对象调用)
针对上述问题的提出,我们先来了解一些必要,且很基础的问题。
结论:实际参数是调用有参方法时真正传递的内容,而形式参数是用于接收实参内容的参数。
我们都知道,在Java中定义方法时是可以定义参数的。比如 Java 中的 main
方法,public static void main(String[] arsg)
, 其中args
就是参数。 参数在程序语言中分为形式参数和实际参数。
当调用方法时,需要把实际参数传递给形式参数, 在传递的过程中到底传递的是什么?
这其实就是程序设计中 求值策略的概念。
按照如何处理传递给函数的实际参数, 求职策略分为:严格求值 和 非严格求值
在函数调用过程中,传递给函数的实际参数总是在应用这个函数之前进行求值。
多数现存编程语言对函数都是使用严格求值策略。
在严格求值中,有几个比较关键的求值策略是我们比较关系的:
(call by value)
-- 值传递
(call by refrence)
-- 引用传递
(call by sharing)
-- 共享对象传递
其实我们会发现, 传值引用和传共享对象引用的调用过程几乎是一样的,都是进行 求值 -- 复制 -- 传递
。但是他们的结果又是不一样的,这是为什么呢?对于这个问题,我们更多的应该关注的是过程, 而不是结果。
因为传共享对象调用的过程和传值调用的过程是一样的,而且都有进一步关键的操作 就是复制
,所以通常我们认为 传共享对象引用就是传值引用的一个特例。 (值 -> 共享对象的地址)
传值调用是值在调用过程中将实际参数复制一份并传递到函数中, 传引用调用是值在调用过程中将实际参数的引用直接传递到函数中。
所以传值调用和传引用的调用的主要区别就是 实际参数是直接传递的, 还是传递副本的。
我们前面讲过,无论是值传递还是引用传递, 其实都是系统设计中的求值策略的一种。
在 《The Java Tutorials》 中, 有关这部分的内容, 也做了说明。
关于基本类型的描述 ,大致的意思是:
原始参数通过值传递给方法。这就意味着对参数值的任何改变都只存在于方法的范围内。当方法返回时,参数将消失,对它的任何改变都将丢失。
关于对象的描述,大致意思是:
引用数据类型参数(对象)也是按照值传递给方法, 这意味着, 当方法返回时,传入的引用仍然引用与以前相同的对象。但是,如果对象字段具有适当的访问级别, 则可以在方法中更改这些字段的值。
上面的引言是官方指南的说明,Java就是值传递,只不过是把对象的引用(地址)当做值传递给方法。这就是我们前面所说的 传共享对象调用。
其实Java中使用的求值策略就是传共享对象调用,也就是说,Java会将对象的地址的副本传递给被调用函数的形式参数。只不过“传共享对象调用” 这个词并不常用,所以Java 社区的人通常说这是 传值调用。这么说也没错, 因为传共享对象调用其实也就是传值调用的一个特例而已。
public class User { private String name; private String male; public User() { } public User(String name, String male) { this.name = name; this.male = male; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getMale() { return male; } public void setMale(String male) { this.male = male; } @Override public String toString() { return "User{" + "name='" + name + '\'' + ", male='" + male + '\'' + '}'; } }
public class PassTest { public static void main(String[] args) { PassTest passTest = new PassTest(); User qzk = new User(); qzk.setName("qzk"); qzk.setMale("男"); passTest.pass(qzk); System.out.println("print in main , user : "+qzk); } public void pass(User user){ user.setName("qian zheng kai"); System.out.println("print in pass , user : " + user); } }
在参数传递的过程中,实际参数的地址 0x123456 被赋值给了形参。 这个过程其实就是值传递,只不过传递的值的内容是对象的引用。
也就是说,Java对象的传递是通过复制的方式吧引用传递了,如果我们没有修改引用关系,而是找到引用的地址, 把里面的内容修改了,则会对调用方有影响,因为形参和实参指向的是同一个共享对象。
小结: