美文网首页
java Comparable与Comparator的区别

java Comparable与Comparator的区别

作者: 赵仝 | 来源:发表于2017-02-08 11:46 被阅读0次

    java中我们要实现集合元素的比较和排序有两种方法。一种是让集合中的元素去实现Comparable接口,另一种是则是创建一个Comparator比较器(创建比较器就是实现Comparator接口)。第一种我们是在集合内部实现的,因为我们让集合中的元素去实现Comparable接口,我们改变了元素的结构。第二种,我们是在集合外部去实现的。

    • 实现Comparable接口,首先我们来看看Comparable接口有什么用,我们为什么要去实现它,这里我觉得最好的说明就是jdk源码中的这段注释。

    Compares this object with the specified object for order. Returns a
    negative integer, zero, or a positive integer as this object is less
    than, equal to, or greater than the specified object.

    这段话的主要意思就是说,Comparable接口是为了将此元素与指定元素排序。将返回正数,零,负数,分别表示大于,等于,小于指定元素的含义。

    下面就是jdk中Comparable接口的唯一一个方法。

         * @param   o the object to be compared.
         * @return  a negative integer, zero, or a positive integer as this object
         *          is less than, equal to, or greater than the specified object.
         *
         * @throws NullPointerException if the specified object is null
         * @throws ClassCastException if the specified object's type prevents it
         *         from being compared to this object.
         */
        public int compareTo(T o);
    

    当我们使用时只需要实现此接口,并实现CompareTo()方法。

    class Student1 implements Comparable<Student1> {
        
        private int id;
        @Override
        public int compareTo(Student1 o) {
            
            
            return o.id>this.id?1:(o.id==this.id?0:-1);
        }
        
    }
    

    这里我采用了泛型的写法,在java中使用泛型的好处,我就引用百度百科的这段话加以说明。

    泛型是Java SE 1.5的新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法。 Java语言引入泛型的好处是安全简单。
    在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的“任意化”,“任意化”带来的缺点是要做显式的强制类型转换,而这种转换是要求开发者对实际参数类型可以预知的情况下进行的。对于强制类型转换错误的情况,编译器可能不提示错误,在运行的时候才出现异常,这是一个安全隐患。
    泛型的好处是在编译的时候检查类型安全,并且所有的强制转换都是自动和隐式的,以提高代码的重用率。

    • 使用Comparator比较器。其实就是去实现Comparator接口。jdk源码中注释如下。

    Compares its two arguments for order. Returns a negative integer,
    zero, or a positive integer as the first argument is less than, equal
    to, or greater than the second.<p>

    让两个元素有序,返回负整数,零,正整数表示第一个参数小于,等于,大于第二个元素。这里说的两个参数,实际上是指Comparator接口中的这个方法的两个参数。

           @param o1 the first object to be compared.
         * @param o2 the second object to be compared.
         * @return a negative integer, zero, or a positive integer as the
         *         first argument is less than, equal to, or greater than the
         *         second.
         * @throws NullPointerException if an argument is null and this
         *         comparator does not permit null arguments
         * @throws ClassCastException if the arguments' types prevent them from
         *         being compared by this comparator.
         */
        int compare(T o1, T o2);
    

    代码一:

    class Student2 implements Comparator<Student2>{
        private int id;
        
        public int getId() {
            return id;
        }
    
        public void setId(int id) {
            this.id = id;
        }
    
        @Override
        public int compare(Student2 o1, Student2 o2) {
            
             return o1.id>o2.id?1:(o1.id==o2.id?0:-1);
        }
        
    }
    

    代码二:

    class MyComparator implements Comparator<Student2>{
    
        @Override
        public int compare(Student2 o1, Student2 o2) {
        
            return o1.getId()>o2.getId()?1:(o1.getId()==o2.getId()?0:-1);
        }
        
    }
    

    我们经常认为代码一写法和实现Comparable没啥区别,但仔细看来还是有很大区别。首先我们代码一中的方法是compare与compareTo()是不一样,其次我们需要传入两个指定元素。最后虽然我们写在Student2 的类中,但与Student2类没有多大关系,我们还可以在方法中去比较Student1。
    代码三

    class Student2 implements Comparator {
        private int id;
    
        public int getId() {
            return id;
        }
    
        public void setId(int id) {
            this.id = id;
        }
    
        @Override
        public int compare(Object o1, Object o2) {
            if (o1 instanceof Student1 && o2 instanceof Student1) {
                return ((Student1) o1).getId() > ((Student1) o2).getId() ? 1
                        : (((Student1) o1).getId() == ((Student1) o2).getId() ? 0
                                : -1);
            } else if (o1 instanceof Student2 && o2 instanceof Student2) {
                return ((Student2) o1).getId() > ((Student2) o2).getId() ? 1
                        : (((Student2) o1).getId() == ((Student2) o2).getId() ? 0
                                : -1);
            } else
                return -1;
    
        }
    
    }
    

    其实Comparator 采用的是策略设计模式,即我会根据传入对象的不同类型,来采用不同的比较策略。当然我们加上泛型之后,也就是只为当前元素服务了。所以如果你的比较器只是为一种元素做比较,你可以按照代码一的写法去写,如果你想做一个通用的比较器,那就按照代码二去写,分离出来,让你的程序的很有可读性。第三种只是为了解决大家的疑惑,开发中应该杜绝这种写法。

    相关文章

      网友评论

          本文标题:java Comparable与Comparator的区别

          本文链接:https://www.haomeiwen.com/subject/akevittx.html