详解java关于对象的比较

目录
  • 同类型对象的比较
    • 样例引入
    • 相等性判断
  • 总结

同类型对象的比较

三个维度去比较

同一性

相等性

相似性

样例引入

想象一下这样的一个场景:小王去图书馆借了一本java核心技术卷1,如图

不幸的是小王把书丢了,他又买了两本java核心技术卷1

                   

新买的书1                    新买的书2

若小王用新买的书1去还书

1,若图书馆禁止他还书  //即不具备同一性(不是同一本书)

2,若图书馆允许他还书  //具备“相等性”(不是同一本,只要相等即可)

若小王用新买的书2去还书

3,若图书馆允许他还书   //具备相似性

Book b1 = new Book("白皮的java核心卷1");        b1和b1之间 <-> 同一性关系

Book b2 = new Book("白皮的java核心卷1");        o1==o2 <-- 是判断什么关系,判断的是同一性,

Book b3 = new Book("黄皮的java核心卷1");         即o1和o2是否指向同一个对象。

b1 == b1 =>true

b1 == b2 =>false

b1 == b3 =>false

相等性判断

java中没有原生提供相似性判断的逻辑,但有相等性判断的约束,即equals(相等)

来自Object类中有一个equals方法。+ 所有类都继承自Object。 =>所有类都有自己的equals方法

java希望类的实现者(谁定义的类),去书写正确的equals方法来保证相等性

希望:

b1.equals(b1) =>true

b1.equals(b3) => false

b1.equals (b2) =>true

public class Main {
    public static void main(String[] args) {
        Book b1 = new Book("白色","java核心卷1");
        Book b2 = new Book("白色","java核心卷1");
        Book b3 = new Book("黄色","java核心卷1");
        //查看同一性
        System.out.println(b1==b1);//true
        System.out.println(b2==b2);//true
        System.out.println(b3==b3);//true
        System.out.println(b1==b2);//false
        System.out.println(b1==b3);//false

        //查看相等性
        System.out.println(b1.equals(b1));//true
        System.out.println(b1.equals(b2));//希望是true//实际上打印false
        System.out.println(b1.equals(b3));//false
    }
}

上例中b1.equals(b2)为false,原因是Book中没有重写equals,故执行了它的父类Object的equals方法,而在Object中,equals还是在判断同一性,所以结果自然显示false。

方法若想b1.equals(b2)为true,若需要正确的重写Book的equals方法。

重写方法后必须保证正确性

什么是正确性:

  • 自反性 =>b1.equals(b1) 得是true
  • 当b1.equals(b2)=> true ;b2.equals(b2) =>true
  • 传递性
    • b1.equals(b2) => true && b2.equals(b3) =>true 可得b1.equals(b3) =>true
  • 4,和null的结果一般是false
    • b1.equals(null) => false

这里不需要自己手动写,使用工具生成正确的equals即可

Code -> Generate ->equals();

public class Book {
    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;
        Book book = (Book) o;
        return color.equals(book.color) &&
                name.equals(book.name);
    }

    public String color;
    public String name;

    public Book(String color ,String name) {
        this.color = color;
        this.name = name;

    }

重写了equals方法后,再去运行,此时 已经重写了equals方法,b1.equals(b2)为true。

总结

同一性,相等性,相似性(相似性java不支持)

同一性和相等性

==比较:同一性比较(对于基本类型,同一性就是相等性;对于引用类型,同一性只是同一性)

要正确处理相等性

1,b1.equals(b2)    2,Book正确的重写了equals方法

满足几个性质:(上边列出的正确性的几条性质)

利用idea工具生成

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

(0)

相关推荐

  • 浅谈java对象的比较

    目录 1.元素的比较 2.类的比较 3.比较方法 3.1 重写equals方法 3.2 基于Comparble接口类的比较 3.3 基于比较器比较基于比较器比较:Comparator接口 3.4 三种比较方式的对比 1.元素的比较 在java中,基本类型的对象可以直接比较大小. public static void main(String[] args) { int a=12; int b=55; System.out.println(a > b); System.out.println(a =

  • 在Java中如何比较两个对象浅析

    Common Lang 中的 Builder 包内提供了一个 DiffBuilder 类,可以比较两个对象,并返回不同的部分. 首先在要比较对象的类中实现 Diffable 接口,然后实现 DiffResult diff(T obj)  方法. 在DiffResult diff(T obj)  方法中,新建一个 DiffBuilder 对象,把需要比较的类属性一一放入 DiffBuilder 中. DiffBuilder 的构造函数有三个入参,lhs 是当前对象,rhs 是要比较的对象,styl

  • Java各种比较对象的方式的对比总结

    一.==和!=操作符 让我们从==和!=开始可以分别判断两个Java对象是否相同的操作符. 1.1 原始类型(Primitives) 对于原始类型,相同意味着具有相等的值: assertThat(1 == 1).isTrue(); 感谢自动拆箱,在将原语值与其包装类型对应值进行比较时,也可以这样做: Integer a = new Integer(1); assertThat(1 == a).isTrue(); 如果两个整数的值不同,==运算符将返回false,而!=运算符将返回true. 1.

  • Java如何比较两个对象并获取不相等的字段详解

    目录  写在前面 缘起 实现 使用方法 扩展 后记  写在前面 在工作中,我们经常会遇到这样的需求--比较两个对象是否相等,如果不相等的话,取出不相等的字段. 以下这些场景都需要我们对一个对象进行比较: 数据比对 做单元测试断言对象是否相等 前端要求对不相等的字段进行高亮显示 这种需求其实是非常简单的,但是如何优雅地解决这一类需求呢? 通常的做法是重写对象的 equals 方法.但是重写 equals 方法有很多缺点,例如: 每次对象属性有变更,一定要记得再重写(放心,你一定会忘记的) 每个对象

  • 基于java中两个对象属性的比较

    两个对象进行比较相等,有两种做法: 1.情况一:当仅仅只是判断两个对象是否相等时,只需重写equals()方法即可.这里就不用说明 2.情况二:当除了情况一之外,还需知道是那个属性不同,那么就需要采用类反射, 具体代码如下: public static void main(String[] args) { A a = new A(); a.setUserName("a"); a.setPassword("p"); a.setQq("q"); a.

  • Java比较对象大小两种常用方法

    引入原因: Java中的对象,正常情况下,只能进行比较:== 或!= ,不能使用 < 或 > ,但是在开发时需要用到比较对象的大小 1.Comparable接口的使用(自然排序) 1.像String .包装类等实现了Comparable接口,重写了compareTo()方法,给出了比较两个对象大小的方法 2.像String .包装类等重写了compareTo()方法后,默认执行了从小到大的排序 3.重写compareTo()的规则: 如果当前对象this大于形参对象obj,则返回正整数,如果当

  • 详解Java中对象序列化与反序列化

    序列化 (Serialization)是将对象的状态信息转换为可以存储或传输的形式的过程.一般将一个对象存储至一个储存媒介,例如档案或是记亿体缓冲等.在网络传输过程中,可以是字节或是XML等格式.而字节的或XML编码格式可以还原完全相等的对象.这个相反的过程又称为反序列化. Java对象的序列化与反序列化 在Java中,我们可以通过多种方式来创建对象,并且只要对象没有被回收我们都可以复用该对象.但是,我们创建出来的这些Java对象都是存在于JVM的堆内存中的.只有JVM处于运行状态的时候,这些对

  • 详解Java多态对象的类型转换与动态绑定

    Java多态对象的类型转换 这里所说的对象类型转换,是指存在继承关系的对象,不是任意类型的对象.当对不存在继承关系的对象进行强制类型转换时,java 运行时将抛出 java.lang.ClassCastException 异常. 在继承链中,我们将子类向父类转换称为"向上转型",将父类向子类转换称为"向下转型". 很多时候,我们会将变量定义为父类的类型,却引用子类的对象,这个过程就是向上转型.程序运行时通过动态绑定来实现对子类方法的调用,也就是多态性. 然而有些时候

  • 详解java关于对象的比较

    目录 同类型对象的比较 样例引入 相等性判断 总结 同类型对象的比较 三个维度去比较 同一性 相等性 相似性 样例引入 想象一下这样的一个场景:小王去图书馆借了一本java核心技术卷1,如图 不幸的是小王把书丢了,他又买了两本java核心技术卷1                     新买的书1                    新买的书2 若小王用新买的书1去还书 1,若图书馆禁止他还书  //即不具备同一性(不是同一本书) 2,若图书馆允许他还书  //具备“相等性”(不是同一本,只要

  • 详解Java对象的强、软、弱和虚引用+ReferenceQueue

    详解Java对象的强.软.弱和虚引用+ReferenceQueue 一.强引用(StrongReference) 强引用是使用最普遍的引用.如果一个对象具有强引用,那垃圾回收器绝不会回收它.当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠随意回收具有强引用的对象来解决内存不足的问题. 二.软引用(SoftReference) 如果一个对象只具有软引用,则内存空间足够,垃圾回收器就不会回收它:如果内存空间不足了,就会回收这些对象的内存.只要垃圾回

  • 详解Java中的不可变对象

    不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真正意图和考虑点是什么?可能一些朋友没有细想过这些问题,今天我们就来聊聊跟不可变对象有关的话题. 一.什么是不可变对象 下面是<Effective Java>这本书对于不可变对象的定义: 不可变对象(Immutable Object):对象一旦被创建后,对象所有的状态及属性在其生命周期内不会发生任何变化. 从不可变对象的定义来看,

  • 详解java 对象锁与类锁

    一.什么是对象锁 对象锁也叫方法锁,是针对一个对象实例的,它只在该对象的某个内存位置声明一个标识该对象是否拥有锁,所有它只会锁住当前的对象,而并不会对其他对象实例的锁产生任何影响,不同对象访问同一个被synchronized修饰的方法的时候不会阻塞, 例如: public class MyObject { private synchronized void method1(){ try { System.out.println(Thread.currentThread().getName());

  • 详解Java对象序列化为什么要使用SerialversionUID

    1.首先谈谈为什么要序列化对象 - 把对象转换为字节序列的过程称为对象的序列化. - 把字节序列恢复为对象的过程称为对象的反序列化. 对象的序列化主要有两种用途: 1) 把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中: 2) 在网络上传送对象的字节序列. 在很多应用中,需要对某些对象进行序列化,让它们离开内存空间,入住物理硬盘,以便长期保存.比如最常见的是Web服务器中的Session对象,当有 10万用户并发访问,就有可能出现10万个Session对象,内存可能吃不消,于是Web容器

  • 详解Java对象的内存布局

    前言 今天来讲些抽象的东西 -- 对象头,因为我在学习的过程中发现很多地方都关联到了对象头的知识点,例如JDK中的 synchronized锁优化 和 JVM 中对象年龄升级等等.要深入理解这些知识的原理,了解对象头的概念很有必要,而且可以为后面分享 synchronized 原理和 JVM 知识的时候做准备. 对象内存构成 Java 中通过 new 关键字创建一个类的实例对象,对象存于内存的堆中并给其分配一个内存地址,那么是否想过如下这些问题: 这个实例对象是以怎样的形态存在内存中的? 一个O

  • 详解Java对象创建的过程及内存布局

    一.对象的内存布局 对象头 对象头主要保存对象自身的运行时数据和用于指定该对象属于哪个类的类型指针. 实例数据 保存对象的有效数据,例如对象的字段信息,其中包括从父类继承下来的. 对齐填充 对齐填充不是必须存在的,没有特别的含义,只起到一个占位符的作用. 二.对象的创建过程 实例化一个类的对象的过程是一个典型的递归过程. 在准备实例化一个类的对象前,首先准备实例化该类的父类,如果该类的父类还有父类,那么准备实例化该类的父类的父类,依次递归直到递归到Object类. 此时,首先实例化Object类

  • 详解Java内部类与对象的打印概念和流程

    目录 一.内部类的概念 二.内部类的分类 三.成员内部类 1.普通内部类 2.静态内部类 四.局部内部类 五.对象的打印 一.内部类的概念 在 Java 中,可以将一个类定义在另一个类或者一个方法的内部,前者称为内部类,后者称为外部类.内部类也是封装的一种体现. public class OutClass {//外部类 class InnerClass{//内部类 } } 注意事项: 1.内部类一定是定义在class 类名{}之中的类,定义在class 类名{}之外的,哪怕是在一份文件中,也并不

随机推荐