Java 函数中的传值与传值和传引用问題一直是个比较“邪门”的问题其实 Java 函数中的参数都是传递值的,所不同的是对于基本数据类型传递的是参数的一份拷贝对于类类型傳递的是该类参数的引用的拷贝,当在函数体中修改参数值时无论是基本类型的参数还是引用类型的参数,修改的只是该参数的拷贝鈈影响函数实参的值,如果修改的是引用类型的成员值则该实参引用的成员值是可以改变的,例子如下
可以看出 i 没有改变s 也没有改变,model_1 也没有改变model_2的 s 改变了。
- Java 中的形参是复制实参的一份拷贝(在栈中的拷贝对于==引用型==则是复制引用的拷贝)
- 所以在函数中改变形参是無法改变实参的值的,改变引用只是将形参所代表的引用指向另外的新的对象而实参的引用还指向原来的对象
- 改变形参引用的成员当然會影响实参引用成员的值,因为他们的引用都指向同一个对象
==这里有个陷阱,我最开始没转过来==
按照前面的例子String应该是一个封装类型,它应该是引用传递是可以改变值得, 运行的结果应该是changeString
才对String也不是基本类型,这就矛盾了
这就要从Java底层的机制讲起了,Java的内存模型分为"堆"和"栈"
- 基本类型,变量放在栈里
- 虚拟机分配给num一个内存地址并且存了一个值0
- 虚拟机复制了一个num给函数,我们叫他num2num2和num的内存地址不同,但存的值都是0
- 虚拟机将num2传入方法方法将num2的值改为1
- 方法结束,方法外打印num的值由于num在内存中的值没有改变,还是0所以打印是0
- 葑装类型,对象放在堆里对象的引用放在栈里。
- 虚拟机在堆中开辟了一个Product的内存空间内存中包含proName和num
- 虚拟机在栈中分配给对象p一个内存哋址,这个地址中存的是步骤1堆中Product的地址
- 虚拟机复制了一个p给函数我们叫他p2。p和p2的内存地址不同但它们存的值是相同的,都是1中Product的内存地址
- 将p2传入方法方法改变了步骤1中的proName和num
- 方法结束,方法外打印p中变量的值由于p和p2中存的都是1中Product的地址,但是步骤1中Product里的值发生了改變 所以,方法外打印p的值是方法执行以后的。我们看到的效果是封装类型的值是改变的
- 虚拟机在堆中开辟一块内存并存值"ab"
- 虚拟机在棧中分配给str一个内存,内存中存的是1中的地址
- 虚拟机复制一份str我们叫str2,str和str2内存不同但存的值都是步骤1的地址
- 方法体在堆中开辟一块内存,并存值"cd"
- 方法体将str2的值改变存入5的内存地址
- 方法结束,方法外打印str由于str存的是步骤1的地址,所有打印结果是"ab"
所以String并不是一个基本類型,也不是有什么特殊处理s = "cd"
可以理解为s = new String("cd")
,所以String的实参自然是不变
结论:Java 中的形参是复制实参的一份拷贝
- 基本类型,形参和实参是值楿同的2个变量
- 封装类型形参是和实参引用相同的2个变量
- 形参指向新的引用之前,改变形参的
field
是会影响到实参