Java形参、实参、值传递、引用传递
转载于https://www.cnblogs.com/wudiffs/p/11573314.html——原作者:wudiffs
一、形参
形参:用来接收调用方法时传递的参数,只有在被调用时才分配内存,一旦调用结束,就释放内存空间。因此仅在方法内有效。
public void swap(int a, int b) { int temp = a; a = b; b = temp; }
这里a,b就是形参,方法结束时,在栈空间就会被销毁。
二、实参
调用上面方法swap(6, 8); 其中6 8就是实参。
三、值传递和引用传递
- 值传递:方法调用时,实际参数将它的值传递给形式参数,函数就收到的是原始值的副本,此时内存中存在两个相同的基本类型,若方法中对形参执行处理操作,并不会影响实际参数的值。
- 引用传递:方法调用时,实际参数的引用(指内存地址,不是参数的值)被传递给方法中相应的形式参数,函数接受到的是原始值的内存地址,在方法中,形参和实参的内容(地址)相同,方法中对形参的处理会影响实参的值。
public class People { private String name; private String age; public People(String name, String age) { this.name = name; this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getAge() { return age; } public void setAge(String age) { this.age = age; } } public class Base1 { public static void main(String[] args) { int a = 10; // 按值传递 test1(a); System.out.println("main a==" + a); People p = new People("rose", "18"); // 按引用传递 test2(p); System.out.println("main name==" + p.getName() +", age==" + p.getAge()); } public static void test1(int a) { a = 20; System.out.println("test1 a==" + a); } public static void test2(People p) { p.setName("jack"); p.setAge("19"); System.out.println("test2 name==" + p.getName() +", age==" + p.getAge()); } }
输出结果: test1 a==20 main a==10 test2 name==jack, age==19 main name==jack, age==19
结论:
- 形参为基本类型时,对形参的处理不会影响实参。
- 形参为应用类型时,对形参的处理会影响实参。
- String、Integer、Double等immutable类型,因为本身没有提供修改函数的操作,每次操作都是生成一个新对象,所以要特殊对待,可以理解为值传递,形参操作不会影响实参对象。
四、java对象及引用
以上面的People为例,构建一个对象:People p1 = new People("jack", "19");
其实这个对象包含四个动作。
- 右边的new People,是以People类为模板,在堆空间创建一个People类对象。
- 末尾的("jack", "19"),指的是对象创建后,调用的构造函数,对刚生成的对象进行初始化。
- 左边的People p1创建了一个People类引用变量,所谓People类引用,就是以后可以用来指向People对象的引用。
- = 操作符使对象引用指向刚创建的People对象。
People p1; p1 = new People("jack", "19"); 上面两条语句等价于 People p1 = new People("jack", "19");
这样看就很明白,一个是对象应用变量,一个是对象实体。
如果只执行第一条,还没执行第二条此时创建的p1还没指向任何一个对象,它的值是null。
再来一句: People p2; p2 = p1; 这里发生了复制行为,对象本身没有复制,被复制的只是对象的引用。结果是p2也指向了p1的对象。
再来一句: p2 = new People("rose", "18"); 此时p2指向了第二个对象,此时得出结论: 一个对象引用可以指向0个或者1个对象。 一个对象可以有N个引用指向它。
再来一句: p1 = p2; 此时p1指向p2这个对象,至于之前的p1这个对象,它已成为垃圾回收机制的处理对象。
五、java中方法参数的引用传递
“当一个对象被当作参数传递到一个方法”,这就是所谓的引用传递。
public class Base2 { public static void main(String[] args) { People p = new People("jordon","23"); // changePeople1(p); changePeople2(p); System.out.println(p.getName() + "===" + p.getAge()); } public static void changePeople1(People p) { p.setName("kobe"); p.setAge("24"); } public static void changePeople2(People p) { p.setName("kobe"); p.setAge("24"); p = new People("james", "24"); } } 以上代码调用changePeople1和changePeople2都是输出 kobe===24 我们来分析下为什么了??