Java中形参,实参,值传递,引用传递
学习值传递,和引用传递很容易令人迷糊,那么就来仔细的分析一下。
想明白这个问题首先要知道形参和实参
实参:就是我们要传递给方法的实际参数
形参:就是我们方法签名上定义的参数
形参和实参在代码上的位置
public class A {
public static void example(int a){//形参
System.out.println(a+1);
}
public static void main(String[] args) {
int a=10;//实参
System.out.println(a);
example(a);
System.out.println(a);
}
}
从这里我们可以看到,形参在方法外边,实参在方法里面
掌握了形参和实参,我们看值传递和引用传递
值传递:对形参的修改不会影响到实参
引用传递:对形参的修改会影响实参
public class A {
public static void example(int a){//形参
a=20;
System.out.println("形参:"+a);
}
public static void main(String[] args) {
int a=10;//实参
System.out.println("实参:"+a);
example(a);//调用带有形参的方法
System.out.println("实参"+a);
}
}
输出结果:
在基本类型的参数传递中我们可以发现:
基本类型当中,虽然形参修改后相关方法的值改变了,但是并没有影响到后序实参打印出来的值。
由此我们可以判断,在Java中基本类型使用的是值传递
接下来是引用类型
public class Person {
public String name;
public Person(String name){
this.name=name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public static void main(String[] args) {
Person p1=new Person("张三");
System.out.println(p1.getName());
p1.setName("李四");
System.out.println(p1.getName());
}
}
输出结果:
在这里我们可以看到,在引用类型中将对象传递给方法中,在方法中改变形参的值,最后输出形参的值也会发生改变。
但是,这并不等于在引用类型中使用的是引用传递!
我们接着看
public class Person {
public String name;
public Person(String name){
this.name=name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public static void change(){
Person p1=new Person("李四");
System.out.println(p1.getName());
}
public static void main(String[] args) {
Person p1=new Person("张三");
System.out.println(p1.getName());
change();
System.out.println(p1.getName());
}
输出结果:
这时需要了解JVM基本知识
在JVM中,栈存储对象,而堆存储基本数据类型和局部变量
基本类型的参数传递过程中
我们可以发现,在操作带有形参的方法时,实际上相当于实参复制一个副本传递给形参,这样在修改形参方法之后,是不会影响到实参的。
而引用类型中
我们可以看到,当形参方法适用对象时,实际上是实参复制的一个副本,这个副本和主体一样,都可以操作堆中的内容。
因此,当调用形参方法中对堆中内容进行修改时,main方法中所指向的值也会改变。
实际上修改的是堆中的内容而并非修改了实参
而我们对自定义方法中,重新对对象开堆,那么实际上副本就引用了另一个堆的内容。
也就变成了两个对象引用两个堆,对其中一个进行修改自然不会影响另一个
结论:
Java当中只有值传递,没有引用传递
在基本类型进行参数传递的过程中,是将值复制一份传递给形参
如果是引用类型,就将引用复制一份传递给形参
无论如何,都无法通过形参改变实参