【JavaSE】java对象的比较

元素的比较

基本类型的比较

Java中,基本类型可以直接进行大小的比较

        //1.基本元素的比较
        int a=10;
        int b=20;
        System.out.println(a>b);
        System.out.println(a<b);
        System.out.println(a==b);
        System.out.println("=============================");
        char ch1='A';
        char ch2='a';
        System.out.println(ch1==ch2);
        System.out.println(ch1>ch2);
        System.out.println(ch1<ch2);
        System.out.println("==========================");
        boolean b1=true;
        boolean b2=false;
        System.out.println(b1==b2);
        System.out.println(b1!=b2);

运行结果:
在这里插入图片描述

对象的比较

		//2.对象的比较
        Student s1=new Student();
        Student s2=new Student();
        Student s3=s2;
        System.out.println(s1==s2);
        //System.out.println(s1<s2);//编译报错
        System.out.println(s3==s2);

运行结果:
在这里插入图片描述

结论:Java的引用类型不能直接进行>或者<的比较.那么为什么可以进行==的比较呢?
因为:对于用户实现自定义类型,都默认继承自Object类,而Object类中提供了equals方法,而==默认情况下调用的就是equals方法.
equal的比较原则是:默认情况下,直接比较
引用变量的地址**

如何进行对象比较

重写equals方法

这里在Student类中重写equals方法

class Student{
    String name;//姓名
    Integer age;//年龄

    public Student(String name,Integer age){
          this.name=name;
          this.age=age;
    }
	//重写equals方法
    @Override
    public boolean equals(Object o) {
        
        if(o==null || !(o instanceof Student)) return false;

            Student obj=(Student)o;

            return this.name.equals(obj.name)?true:false;
    }

}
		//2.对象的比较
        Student s1=new Student("zhangsan",11);
        Student s2=new Student("zhangsan",12);

        Student s3=s2;
        System.out.println(s1==s2);
        System.out.println(s1.equals(s2));
        //System.out.println(s1<s2); ------编译报错
        System.out.println(s3==s2);

重写equals方法虽然可以进行比较,但是只能进行相等比较,不能按照大于,小于的方式进行比较

基于Comparble.compareTo的比较

在类的内部重写compareTo方法:

class Student implements Comparable<Student>{
    String name;//姓名
    Integer age;//年龄

    public Student(String name,Integer age){
          this.name=name;
          this.age=age;
    }
    
    @Override
    public int compareTo(Student o) {
        if(o==null) return 1;
        return age-o.age;
    }
}

进行比较:

public class CompareTest {
    public static void main(String[] args) {
    	Student s1=new Student("zhangsan",11);
        Student s2=new Student("lisi",16);
        Student s3=new Student("zhangsan",12);
		
		System.out.println(s1.compareTo(s2));
        System.out.println(s2.compareTo(s1));
        System.out.println(s3.compareTo(s2));
}
}

运行结果:
在这里插入图片描述

基于Comparator.compare的比较

在类的外部构造一个比较器Comparator:

/**
 * 在外部构造一个比较器
 * */
class StudentComparator implements Comparator<Student>{
    @Override
    public int compare(Student s1, Student s2) {
            return s1.age-s2.age;
    }
}
class Student implements Comparable<Student>{
    String name;//姓名
    Integer age;//年龄

    public Student(String name,Integer age){
          this.name=name;
          this.age=age;
    }
    
}    

调用比较器进行比较

		Student s1=new Student("zhangsan",11);
        Student s2=new Student("lisi",16);
        Student s3=new Student("zhangsan",12);
        //调用比较器
        StudentComparator stuComparator=new StudentComparator();
        System.out.println(stuComparator.compare(s1, s2));

运行结果:
在这里插入图片描述

区分ComparatorComparable

ComparatorComparable都是Java用于比较的对象,但它们的作用场景和使用方式有所不同:

  • Comparable接口:

    • 它是一个内置接口,通常由类自身实现,用于提供自定义类型的自然顺序。如果一个类实现了Comparable接口,那么它的实例就可以直接通过compareTo()方法与其他同类实例进行比较。比如,String类就是实现了Comparable<String>,可以直接比较两个字符串的大小。
    • 所以,Compareable接口需要手动实现,且代码的侵入性比较强,一旦实现,每次用该类都有顺序,属于内部顺序
  • Comparator接口:

    • 它是另一个独立于对象本身的接口,它允许你在运行时动态地改变比较规则。当需要对列表或其他集合进行排序,但不想修改类本身或者不知道对象内部如何排序时,可以使用Comparator
    • 需要实现一个比较器对象,对待比较类的侵入性弱.

PriorityQueue中比较的体现

由于PriorityQueue底层使用的堆结构,因此其内部的元素必须要能够比大小,PriorityQueue采用了两种比较方式的实现:

  • Comparable:是类默认的内部比较方式,如果用户插入自定义类型的对象时,该类对象必须要实现Comparable接口,并且重写了CompareTo方法
  • Comparator:用户也可以选择使用比较器对象,如果用户插入自定义类型对象时,必须要提供一个比较器类,让该类实现Comparator接口并且重写了compare方法

接下来我们通过源码来进行理解:
当我们进行插入操作时:

public boolean offer(E e) {
        if (e == null)
            throw new NullPointerException();
        modCount++;
        int i = size;
        if (i >= queue.length)
            grow(i + 1);
        siftUp(i, e);//向上调整
        size = i + 1;
        return true;
    }

这里我们看到,调用了关键方法siftUp向上调整方法.

private void siftUp(int k, E x) {
        if (comparator != null)
            siftUpUsingComparator(k, x, queue, comparator);
        else
            siftUpComparable(k, x, queue);
    }

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值