【Java】到底是不是同一个对象?

简介: 【Java】到底是不是同一个对象?

自己一直没有完全搞明白java的传值,传引用,传对象的区别,有时候觉得自己搞懂了,一碰到具体例子,又懵了,所以今天来敲代码看一下。


第一版:


    public static void main(String[] args) {
        List<Integer> userListA=new ArrayList<>();
        userListA.add(1);
        List<Integer> userListB=new ArrayList<>();
        userListB=userListA;
        System.out.println(userListA == userListB);
    }


来看下结果:


true


所以这种直接=的情况,两个list其实指向的是堆内的同一个对象。


第二版:


    public static void main(String[] args) {
        List<Integer> userListA=new ArrayList<>();
        userListA.add(1);
        List<Integer> userListB=new ArrayList<>();
        userListB.add(1);
        System.out.println(userListA == userListB);
    }


答案是:


false


当分别赋值,此时两个list指向不同。


这里有个有趣的地方,此时如果我们不比较list,而是比较list中的第一个对象,你会发现,返回结果也是true。


当我们继续将放到list中的值增大,一直到128,如下:


    public static void main(String[] args) {
        List<Integer> userListA=new ArrayList<>();
        userListA.add(128);
        List<Integer> userListB=new ArrayList<>();
        userListB.add(128);
        System.out.println(userListA.get(0) == userListB.get(0));
    }


此时结果就会变为 false,这种情况,是因为常量池的存在。


Byte, Short, Integer, Long, Character, Boolean, Float, Double, 除Float和Double以外, 其它六种都有常量池,,只在大于等于-128并且小于等于127时使用常量池


第三版:


此时我们不往list中放数字,放个对象试试:


    public static void main(String[] args) {
        User user=new User();
        user.setAge(1);
        user.setName("蝈蝈");
        List<User> userListA=new ArrayList<>();
        userListA.add(user);
        List<User> userListB=null;
        userListB=userListA;
        System.out.println(userListA == userListB);
    }


结果当然是:


true


第四版:


看看这个:


    public static void main(String[] args) {
        User user=new User();
        user.setAge(1);
        user.setName("蝈蝈");
        List<User> userListA=new ArrayList<>();
        userListA.add(user);
        List<User> userListB=new ArrayList<>();
        userListB.add(user);
        System.out.println(userListA == userListB);
        System.out.println(userListA.get(0)==userListB.get(0));
    }


结果为:


false
true


第五版:


到现在为止还是挺清晰的,那么addAll的情况如何呢?


    public static void main(String[] args) {
        User user=new User();
        user.setAge(1);
        user.setName("蝈蝈");
        List<User> userListA=new ArrayList<>();
        userListA.add(user);
        List<User> userListB=new ArrayList<>();
        userListB.addAll(userListA);
        System.out.println(userListA == userListB);
        System.out.println(userListA.get(0)==userListB.get(0));
    }


结果:


false
true


总结


其实java中,只有一种参数传递,就是值传递,当一个对象是基本数据类型对象时,把它当作参数传递给另一个方法,此时传递的是值,也就是会讲值拷贝给另一个方法。基本类型数据在方法里被修改了,原来的调用方那里是不会变的,因为是被复制了一份的,不会影响到原来的值;


当一个对象是个对象引用类型,比如User对象时,把它当作参数传递给另一个方法,此时传递的也是值,只不过此时,值并非某个数字,而是堆内存的地址。


目录
相关文章
|
15天前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
19天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
39 17
|
18天前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
27天前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
23 6
|
1月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
18 3
|
1月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
1月前
|
XML Java Maven
在 Cucumber 测试中自动将 Cucumber 数据表映射到 Java 对象
在 Cucumber 测试中自动将 Cucumber 数据表映射到 Java 对象
50 7
|
1月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第3天】Java零基础教学篇,手把手实践教学!
14 1
|
2月前
|
Java
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
本文介绍了Java中抽象类和抽象方法的使用,以及ArrayList的基本操作,包括添加、获取、删除元素和判断列表是否为空。
26 2
java基础(12)抽象类以及抽象方法abstract以及ArrayList对象使用
|
1月前
|
Java 数据安全/隐私保护
java类和对象
java类和对象
23 5