关于java到底是值传递还是引用传递

前言

最近在看关于java的一些东西,但在网上查询资料时发现好多人都关于java到底是值传递还是引用传递有一些看法,现在整理一下,对自己的学习做个记录。

首先,需要先清楚java中的变量类型

java中的变量有两类:primitive主数据类型和引用
其中主数据类型又分为八种:
boolean char byte short int long float double

然后,需要先清楚java中的变量类型的区别

那主数据类型和引用类型有什么区别呢?

int num = 10;
String str = "hello";
关于java到底是值传递还是引用传递_第1张图片
主数据类型和引用类型的区别
  • numint类型,属于主数据类型,该变量的内存空间直接存储了值10
  • str属于引用类型,该变量内存空间存储的是String对象"hello"的地址,实际上也就是引用了这个对象。

接着,需要搞清楚String类型存储在Heap上还是String Pool中

  1. 直接赋值字符串,会引用String Pool中的对象(如果没有就创建),这是在编译期就完成的。
String a = "hello";   
String b = "hello";  
System.out.println(a == b);  //true
关于java到底是值传递还是引用传递_第2张图片
String直接赋值
  1. new String会在垃圾回收堆上创建对象并引用(堆上创建时会从String Pool中深拷贝),这是在运行时完成的。
String a = new String("hello") ;
String b = new String("hello") ;
System.out.println(a == b);  //false
关于java到底是值传递还是引用传递_第3张图片
new产生新的对象

最后,我们以两个传参的例子来讲

//示例一
String a = "hello";
String b = foo(a);
System.out.println(a == b);  //false

String foo(String str) {
    str = str + "world";
    return str;
}

解析:用+做字符串连接时会返回新的String对象,而不是在原有的对象上操作。
foo将实参a传递给形参str,相当于执行了str=a,此时stra都引用了同一个对象,但当执行 str=str+"world"; 后,str引用了+操作符返回的一个新的String对象,值为helloworldb就是str,而a还是引用之前的对象,值为hello,所以结果为false

//示例二
StringBuilder a = new StringBuilder("hello");
StringBuilder b = foo(a);
System.out.println(a == b); //true

StringBuilder foo(StringBuilder builder) {
    builder = builder.append("world");
    return builder;
}

解析:示例二和示例一的区别在于StringBuilder类的append方法不会返回新的String对象,而是在原有String对象上操作,所以结果为trueab的值均为helloworld

关于java到底是值传递还是引用传递_第4张图片
Java API append方法

总结

对于java到底是值传递还是引用传递,很多人有自己的理解,甚至很多人还在争辩值传递和引用传递的定义,我觉得这些字面理解对于初学者都是没有意义的,我们最应该关注的是传递本身到底做了什么事情。

  • 主数据类型不必多说,它是传递了一个值的拷贝,也就是副本,所有的操作与原来的值再无关联。
  • 引用类型的话,我们只需要关注值在传递后有没有生成新的对象,如果生成了新的对象,所有操作与原值无关,如上示例一。如果没有生成新的对象,将会在原值上操作,传递前后的两个变量是一条绳上的蚂蚱,你变我也变,如上示例二。

学习资料:

  • java到底是值传递还是引用传递?--知乎
  • Head First Java 对象引用(54--58), String 不变性(附录B#9) 。

你可能感兴趣的:(关于java到底是值传递还是引用传递)