Java的值传递和引用传递

网络上能搜索到的资料里,关于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 

发表评论

评论已关闭。

相关文章