对象引用与值引用

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

一 基本概念

首先我们要知道

  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都指向了李四。

三 总结

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

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

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

相关文章
|
7月前
|
存储 编译器 C语言
初谈C++:引用-1
初谈C++:引用
69 0
|
7月前
|
存储 安全 编译器
初谈C++:引用-2
初谈C++:引用
62 0
|
存储 安全 编译器
【C++】C++引用(下)
【C++】C++引用(下)
|
5月前
|
安全 C++
|
7月前
|
存储 C++
c++引用
c++引用
54 1
|
7月前
|
设计模式 JavaScript 前端开发
不正确的引用 this
不正确的引用 this
33 0
|
存储 编译器 C语言
C++引用下
C++引用下
112 1
|
C语言 C++
C++引用上
用不是新定义一个变量,而是给已存在的变量取一个别名,译器不会为引用变量开辟内存空间,它和它引用的变量共用同一块内存空间。
76 0
|
人工智能 安全 编译器
[C++: 引用】(二)
[C++: 引用】(二)
99 0
|
编译器 C语言 C++
[C++: 引用】(一)
[C++: 引用】(一)
52 0