网络上能搜索到的资料里,关于Java到底是值传递还是引用传递的讨论是比较多的,也没有一个特别被大家认可的结论。
因为最近一两年转到了Golang的开发,接触到了比较多的指针的玩法,突然对Java的引用传递和值传递又有了一定的兴趣。
但是我无意于讨论Java到底是值传递还是引用传递,我只是记录一下,避免以后开发的时候踩坑。
为了验证,我写了这么一段代码:
private void transInt(int x) { x += 100; } // 这段代码的调用逻辑如下: int x = 10; m.transInt(x); System.out.println("-----------------after int trans----------------"); System.out.println(x);
这段代码不出意外的会打印10,从这一点上看,Java是值传递的,因为传入transInt的是x的一个副本。
不过事情并不会这么简单的结束,上面的例子太特殊了,我使用的是Java提供的基本类型。
换成String类型是否还能如此,这是一个值的验证的问题,所以应该实现这样一段代码:
private void transString(String x) { x += "bar"; } // 这段代码的调用逻辑如下: String str = "foo"; m.transString(str);
这段代码的执行结果是"foo",也就是说非基本类型也是值传递。
事情到了这一步似乎是可以说Java是值传递了,但是事情并不会这么简单的结束,我实现了一个类:
class Solution implements Cloneable { private int age; private String name; public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } @Override protected Object clone() throws CloneNotSupportedException { return (Solution) super.clone(); } }
然后实现了一个方法:
private void transfer(Solution solution) { solution.setAge(12); solution.setName("lee"); } // 调用方法如下: Solution solution = new Solution(); solution.setName("foo"); solution.setAge(20); System.out.println("--------------before transfer------------------------"); System.out.println(solution.getAge()); System.out.println(solution.getName()); m.transfer(solution); System.out.println("--------------after transfer-------------------"); System.out.println(solution.getAge()); System.out.println(solution.getName());
打印结果如下:
--------------before transfer------------------------ 20 foo --------------after transfer------------------- 12 lee
这种结果很明显就是引用传递了。事到如今又可以说Java是引用传递了。
所以不能简单的说Java是值传递还是引用传递,这里的水还是比较深的,如果不是很了解,很可能在代码实现的时候出现一些奇怪的问题。
这一点上我觉得Java偷感比较重,同样的如果是Golang,则比较直接。
比如这段代码:
package main import "fmt" type Student struct { Name string Age int } func transStu(stu Student) { stu.Age = 12 stu.Name = "bar" } func transStuPoint(stu *Student) { stu.Age = 100 stu.Name = "lee" } func main() { stu := &Student{ Name: "foo", Age: 21, } fmt.Printf("%s:%dn", stu.Name, stu.Age) transStu(*stu) fmt.Printf("%s:%dn", stu.Name, stu.Age) transStuPoint(stu) fmt.Printf("%s:%dn", stu.Name, stu.Age) }
如果传入的是值,那么就是值传递,如果传入的是指针,那么就是引用传递,控制权交给程序员,所以这段代码的打印结果就是:
foo:21 foo:21 lee:100
之前学习C语言的时候,就是因为受不了这么灵活的指针而转投Java,但是现在看来,C语言把大部分的控制权交给程序员不失为一种明智的选择。
基本没见过网上有讨论C或者Go是值传递还是引用传递的。
使用Golang有一点好处就是一下子点开了我的C语言。C语言的好处就是把选择权交给程序员,基本上程序员就是程序的王,如果用C语言的话是不存在这种值传递还是引用传递的争议的,比如下面的代码:
#include<stdio.h> int main() { printf("before trans1n"); int a1 = 10; printf("input value is %d, input address is %pn", a1, &a1); trans1(a1); printf("after trans1n"); printf("%dn", a1); int a2 = 11; printf("before trans2n"); printf("input value is %d, input address is %pn", a2, &a2); printf("%dn", a2); trans2(&a2); printf("after trans2n"); printf("%dn", a2); return 0; } // 传入的是一个值 void trans1(int x) { // 这里的打印值应该和main函数里的不一致,因为传入的是一个全新的数 printf("input x value is %d, input address is %pn", x, &x); x += 100; } // 传入的是int型指针 void trans2(int * x) { // 这里应该和外层的打印值是一样的 printf("input x value is %d, input address is %pn", *x, x); // 取值之后,加100, *x += 100; }
打印结果:
before trans1 input value is 10, input address is 0x7ffffcc3c (下一行的地址和上面不同,传给trans1的实际上是一个全新的值) input x value is 10, input address is 0x7ffffcc10 after trans1 10 before trans2 input value is 11, input address is 0x7ffffcc38 11 input x value is 11, input address is 0x7ffffcc38 (传给trans2的实际是一段地址,和上面的打印结果相同) after trans2 111