Java知识巩固(1)—— 参数的传递

Java中的参数传递很多人都弄不明白是怎么回事,有人说是值传递,有人说是引用传递,或者有人说既是值传递也是引用传递。那么今天就来看看,Java中的参数传递,到底是传值,还是传的引用。
先说结论:
Java中方法参数传递方式是按值传递:

  • 如果参数是基本类型,传递的是基本类型的字面量值的拷贝。
  • 如果参数是引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝

对于第一个结论理解很简单,举个例子来说明:

 public class Main {
     
    public static void main(String[] args) {
     
        Person p = new Person();
        int n = 15;
        p.setAge(15);
        System.out.println(p.getAge());//打印出来为15
        n = 20;
        System.out.println(p.getAge());//打印出来为15
    }

}

class Person {
     
    private int age;

    public int getAge() {
     
        return this.age;
    }

    public void setAge(int age) {
     
        this.age = age;
    }
}

从结果可知,修改外部的局部变量n,不影响实例p的age字段,原因是setAge()方法获得的参数,复制了n的值,因此,p.age和局部变量n互不影响。
结论:基本类型参数的传递,是调用方值的复制。双方各自的后续修改,互不影响。
这一点很好理解,没有什么问题。那么对于第二个结论我们来举两个例子来说明。
例子1:

public class Main {
     
    public static void main(String[] args) {
     
        Person p = new Person();
        int n = 15;
        p.setAge(15);
        System.out.println(p.getAge());//打印出来为15
        System.out.println(System.identityHashCode(p));//打印出地址为764977973

        fun(p);
        System.out.println(p.getAge());//打印出来为20
        System.out.println(System.identityHashCode(p));//打印出地址为764977973
    }

    public static void fun(Person person) {
     
        person.setAge(20);
        System.out.println(System.identityHashCode(p));//打印出地址为764977973
    }
}

class Person {
     
    private int age;

    public int getAge() {
     
        return this.age;
    }

    public void setAge(int age) {
     
        this.age = age;
    }
}

例子2:

public class Main {
     
    public static void main(String[] args) {
     
        Person p = new Person();
        int n = 15;
        p.setAge(15);
        System.out.println(p.getAge());//打印出来为15
        System.out.println(System.identityHashCode(p));//打印出地址为764977973

        fun(p);
        System.out.println(p.getAge());//打印出来为15
        System.out.println(System.identityHashCode(p));//打印出地址为764977973
    }

    public static void fun(Person person) {
     
        person = new Person();
        person.setAge(20);
        System.out.println(System.identityHashCode(person));//打印出地址为381259350
    }
}

class Person {
     
    private int age;

    public int getAge() {
     
        return this.age;
    }

    public void setAge(int age) {
     
        this.age = age;
    }
}

先说下程序中对于值传递和引用传递的定义为:

  • 值传递:指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
  • 引用传递:指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。

两者的区别为:

  • 值传递会创建副本,函数中无法改变原始对象
  • 引用传递不创建副本,函数中可以改变对象

那么有人就会说,第一个例子就说明了会改变对象,不就证明了Java是引用传递嘛。
事实真的是这样的吗?其实通过第二个例子可以很好的说明,java中的参数传递是值传递而不是引用传递。
第二个例子中 fun 方法里new了一个新的对象,如果是引用传递,那么原始对象指向的地址应该是新对象的地址,而结果却是原始对象的地址并没有改变。
那么第一例子为什么会改变原始对象呢?

因为在第一个例子中传递的是对地址的拷贝

比方说,小明有一个房子,房子里面放了一个电脑,小明有一把可以进他家房子的钥匙,有一天小明把钥匙备份了一个给了小明的女朋友,小明女朋友拿着钥匙的备份进小明家,看到小明玩电脑非常暴躁,把小明电脑砸了,所以小明家的电脑没了。

这个钥匙的备份,就相当于是对地址的拷贝

对于第二个例子来说:
小明把备份的钥匙给了他女朋友,但是他女朋友却把钥匙换成了隔壁老王家的钥匙,然后他女朋友去了隔壁老王家,他女朋友在隔壁老王家想怎么来就怎么来,所以不会对小明家的电脑造成影响。而小明拿的是自己家的钥匙,只能回自己家了,去不了隔壁老王家。

所以对于引用类型,传递的是该参量所引用的对象在堆中地址值的拷贝

你可能感兴趣的:(Java基础)