对象引用与值引用

简介: 对象引用与值引用的区别

一 基本概念

首先我们要知道

  1. 八大基本数据类型是存在栈里的;
  2. new出来的对象值是存在堆里的,但是这个对象的引用是存在栈里的(如果new的对象是静态的话,对象的引用在方法区里,而不是栈里);
    image.png

  3. 传递原理:一个方法传递的参数如果是基本数据类型,则是对具体值的拷贝;如果是对象数据类型,则是对对象引用地址值的拷贝,而非具体值拷贝。

二 有三种传递情况

2.1 一个方法不能改变传入基本类型的参数值

public class MyTest {
   
   
    public static void main(String[] args) {
   
   
        int a = 1;
        System.out.println("before a:" + a);
        changeA(a);
        System.out.println("after a:" + a);
    }

    private static void changeA(int b) {
   
   
        b = 2;
        System.out.println("current b:" + b);
    }
}

结果:

before a:1
current b:2
after a:1

图解
image.png

说明:
b=2单独开了一片空间,和a没有任何关系,所以改变b是不会对a的值有任何影响的。

2.2 一个方法不能改变传入对象类型的参数的引用地址

示例

public class MyTest {
   
   
    public static void main(String[] args) {
   
   
        Person pa = new Person("张三");
        System.out.println("before pa:" + pa.getName());
        change(pa);
        System.out.println("after pa:" + pa.getName());
    }


    private static void change(Person pb) {
   
   
        Person pc = new Person("李四");
        pb = pc;
        System.out.println("current pc:" + pc.getName());
        System.out.println("current pb:" + pb.getName());
    }

    public static class Person {
   
   
        private String name;

        public Person(String name) {
   
   
            this.name = name;
        }

        public String getName() {
   
   
            return name;
        }

        public void setName(String name) {
   
   
            this.name = name;
        }
    }
}

结果

before pa:张三
current pc:李四
current pb:李四
after pa:张三

图解

pb=pc赋值前:
image.png

pb=pc赋值后:
image.png

说明:
当开始调用change(pa)的时候,实际上是将pa的地址拷贝了一份然后给了pb,所以pa和pb都指向张三;

pc是从新new的,会重新开辟栈区和堆区的空间;
然后经过pb = pc的赋值操作后,实际上是让他们同时指向堆区的李四。

2.3 一个方法能够改变传入对象类型的参数某一个属性

示例

public class MyTest {
   
   
    public static void main(String[] args) {
   
   
        Person pa = new Person("张三");
        System.out.println("before pa:" + pa.getName());
        change(pa);
        System.out.println("after pa:" + pa.getName());
    }


    private static void change(Person pb) {
   
   
        pb.setName("李四");
        System.out.println(pb.getName());
    }

    public static class Person {
   
   
        private String name;

        public Person(String name) {
   
   
            this.name = name;
        }

        public String getName() {
   
   
            return name;
        }

        public void setName(String name) {
   
   
            this.name = name;
        }
    }
}

结果

before pa:张三
李四
after pa:李四

图解

pb.setName("李四")之前:
image.png

pb.setName("李四")之后:
image.png

说明:
调用change(pa)方法后,且调用pb.setName(“李四”)前,实际上是将pa的地址拷贝了一份然后给了pb,所以pa和pb都指向张三;
调用pb.setName(“李四”)后,实际上是将张三变成了李四,pa和pb引用地址不变,所以pa和pb都指向了李四。

三 总结

总结:
①一个方法不能改变传入基本类型的参数值。

②一个方法不能改变传入对象类型的参数的引用地址。

③一个方法能够改变传入对象类型的参数某一个属性。

相关文章
|
存储 安全 编译器
【C++】C++引用(下)
【C++】C++引用(下)
|
存储 前端开发 rax
【C++】C++引用(上)
【C++】C++引用(上)
|
5月前
|
C++
C++引用
C++引用
|
5月前
|
安全 测试技术 C++
C++中的引用
C++中的引用
29 1
|
6月前
|
存储 程序员 C语言
|
存储 编译器 C语言
C++引用下
C++引用下
108 1
|
编译器 C语言 C++
[C++: 引用】(一)
[C++: 引用】(一)
50 0
|
存储 安全 编译器
C++之引用(下)
C++之引用(下)
83 0
|
存储 安全 编译器
【C++】引用(下)
【C++】引用(下)
【C++】引用(下)
|
存储 编译器 C++
【C++】引用(上)
【C++】引用(上)
【C++】引用(上)