前言
最近在看关于java的一些东西,但在网上查询资料时发现好多人都关于java到底是值传递还是引用传递有一些看法,现在整理一下,对自己的学习做个记录。
首先,需要先清楚java中的变量类型
java中的变量有两类:primitive主数据类型和引用
其中主数据类型又分为八种:
boolean char byte short int long float double
然后,需要先清楚java中的变量类型的区别
那主数据类型和引用类型有什么区别呢?
int num = 10;
String str = "hello";
-
num
是int
类型,属于主数据类型,该变量的内存空间直接存储了值10
。 - 而
str
属于引用类型,该变量内存空间存储的是String对象"hello"
的地址,实际上也就是引用了这个对象。
接着,需要搞清楚String类型存储在Heap上还是String Pool中
- 直接赋值字符串,会引用
String Pool
中的对象(如果没有就创建),这是在编译期就完成的。
String a = "hello";
String b = "hello";
System.out.println(a == b); //true
-
new String
会在垃圾回收堆上创建对象并引用(堆上创建时会从String Pool中深拷贝),这是在运行时完成的。
String a = new String("hello") ;
String b = new String("hello") ;
System.out.println(a == b); //false
最后,我们以两个传参的例子来讲
//示例一
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
,此时str
和a
都引用了同一个对象,但当执行 str=str+"world"
; 后,str引用了+
操作符返回的一个新的String
对象,值为helloworld
,b
就是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
对象上操作,所以结果为true
,a
与b
的值均为helloworld
。
总结
对于java到底是值传递还是引用传递,很多人有自己的理解,甚至很多人还在争辩值传递和引用传递的定义,我觉得这些字面理解对于初学者都是没有意义的,我们最应该关注的是传递本身到底做了什么事情。
- 主数据类型不必多说,它是传递了一个值的拷贝,也就是副本,所有的操作与原来的值再无关联。
- 引用类型的话,我们只需要关注值在传递后有没有生成新的对象,如果生成了新的对象,所有操作与原值无关,如上示例一。如果没有生成新的对象,将会在原值上操作,传递前后的两个变量是一条绳上的蚂蚱,你变我也变,如上示例二。
学习资料:
- java到底是值传递还是引用传递?--知乎
- Head First Java 对象引用(54--58), String 不变性(附录B#9) 。