前言
相信大多数学习java的人都曾有过这样的疑惑,java参数传递究竟是怎么一回事?
1.是不是传基本类型就是值传递,传对象就是引用传递啊?
2.为啥在传递引用的方法中,有时形参变化实参跟着变了,有时却没有呢?
3.java只有值传递?
本文就是围绕这几个问题,针对java参数传递进行学习归纳与总结。如有错误,敬请指正。
概念
在开始学习之前,首先得弄明白什么是值传递,什么是引用传递?
值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递(pass by reference)是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
以上概念来自于百度百科,也是网络上的主流说法。
思考
1.根据这个概念,岂不是承认了第一个问题:传基本类型就是值传递,传对象就是引用传递?
因为我们知道,基本类型传递就是值的拷贝,对象传递实际传递的是对象的地址的拷贝。
2.如这概念所言,引用传递在函数中对形参就行修改,那实际参数就会受到影响。为什么对于第二个问题,实际传递地址之后,函数内对形参进行修改,实参也未必受到影响啊?
3.地址的拷贝是否算值的复制从而算值传递呢?这不就对应了网上的java只有值传递的说法吗?
4.读到这里,可能对值传递、引用传递定义产生疑问,再一百度,发现值传递和引用传递其实是一种求值策略。然后又产生一系列的疑问...
所以,我想说的是没必要去扣字眼来纠结于究竟是值传递还是引用传递,这些字面上的歧义是很容易将人引入误区的,理解了本质就行啦
我们来看一下这段测试的代码:
package com.weixin.javalearingtest;
public class Person {
int age;
int height;
Person(int age,int height){
this.age=age;
this.height=height;
}
public static void main(String[] args){
Person alex =new Person(10,10);
alex.setAge(0);
alex.newPerson(alex);
System.out.println("[" + alex.age + "," + alex.height + "]");
Person army =new Person(9,9);
System.out.println("[" + alex + "]");
exchange(alex,army);
System.out.println("[" + alex.age + "," + alex.height + "]"+"[" + army.age + "," + army.height + "]");
}
private void setAge(int age){
this.age = age;
}
private void newPerson(Person a){
a=new Person(2,3);
}
private static void exchange(Person a,Person b){
System.out.println("[" + a + "]");
Person temp = a;
a=b;
b=temp;
temp.setAge(120);
temp=new Person(8,8);
}
}
运行结果:
[0,10]
[com.weixin.javalearingtest.Person@42a57993]
[com.weixin.javalearingtest.Person@42a57993]
[120,10][9,9]
显然,当实参是对象时,传递给形参的实际是实参对象在堆中地址的拷贝,形参调用方法改变自身,则实参也跟着改变,而形参的重引用对实参无影响。
于是嘞,我们的结论就出炉喽。
结论
当传递的参数是基本类型时,传递的是值的拷贝。
当传递的参数是引用类型时,传递的是实参在堆中地址的拷贝。
当参数是基本类型时,方法无法对其改变。(这其实是基本类型不可变性质所决定的)
当参数是引用类型时,可对其进行状态的改变,但不可改变其引用。