Comparable 与 Comparator

Comparable 与 Comparator异同之处

  1. 都是 java 的一个接口,并且都是用来对自定义的 class 进行比较大小的。
  2. Comparable是通用的接口,用户可以实现它来完成自己特定的比较,而Comparator可以看成一种算法的实现,在容器集合 collection 需要比较功能的时候,来指定这个比较器,这可以看出一种设计模式,将算法和数据分离。
  3. Comparable 比较固定,都是和一个具体的类绑定在一起(类似“静态绑定”);而Comparator 则比较灵活,它可以被各个需要比较功能的类使用(类似"动态绑定")。
  4. 一个实现 Comparable 接口的类,表示该类的所有对象之间是可以比较的,存在着一定的顺序性。这时候就可以用 sort 方法来排序了。
  5. Comparator 接口的作业主要有两个:
    • 若某个程序猿在没有考虑 compare 问题时没有实现 Comparable 接口,恰又需要对象比较时候,就可以选着用 Comparator 接口来实现比较算法的排序。
    • 可以进行多种方式排序,例如:升序、降序、Id排序、成绩排序等等

Comparable的实现

  • 以下代码以一个员工类利用 Comparable 接口实现每员工的排序。首先是声明一个 Employee 员工类,并且实现 Comparable 接口,重写 compare 方法(本例子是通过员工工资排序的)。
  • 如果有人不清楚 compare() 函数意义的,可以利用以下代码中我重写的插入排序算法来加以理解。
  • 最后在主方法中声明一个 Employee 的数组,再将所实例化的员工放入数组中。最后调用Arrays.sort(staff) 进行排序,再输出显示。
import java.util.*;  
/*
* 因为要实现对Employee对象的排序,所以在Employee类中要实现Comparable接口,
* 也就是要实现comepareTo()方法
*/
class Employee  implements Comparator<Employee>  
{  
    public Employee(String n, double s)  
        {  
            name = n;  
            salary = s;  
            Random ID =  new Random();  
            id = ID.nextInt( 10000000 );  
        }  
    public int getId()  
    {  
        return id;  
    }  
    public String getName()  
    {  
        return name;  
    }  
    public double getSalary()  
    {  
        return salary;  
    }  
    
    @Override
    public int compare(Employee other1,Employee other2)  
    {  
        if (other1.salary < other2.salary) 
            return - 1 ;  
        if (other1.salary > other2.salary)  
            return 1 ;  
        return 0 ;  
    }  
    private int id;  
    private String name;  
    private double salary;  
}
public class EmployeeSortTest {  
public static void main(String[] args) {  
        Employee[] staff =  new Employee[ 6 ];  
        staff[ 0 ] =  new Employee( "harry Hacker" , 35000 );  
        staff[ 1 ] =  new Employee( "carl cracke" , 25010 );  
        staff[ 2 ] =  new Employee( "tony Tester" , 38200 );  
        staff[ 3 ] =  new Employee( "Michale" , 23000 );
        staff[ 4 ] =  new Employee( "Cindy" , 31100 );
        staff[ 5 ] =  new Employee( "Yancy" , 32000 );
        Arrays.sort(staff); //sort方法可以实现对对象数组排序,但是必须实现 Comparable接口      
        //insertSort(staff);//插入排序算法
        for (Employee e: staff) {
            if (e != null) 
                 System.out.println( "id=" +e.getId()+ "  name=" +e.getName()+  
                         ".salary=" +e.getSalary());
        }  
    }
    //重写排序算法,利用插入排序。一般使用的 Arrays.sort() 利用的是快速排序算法
    /*public static <AnyType extends Comparable<? super AnyType>>
        AnyType[] insertSort (AnyType arr[]) {//一样实现Comparable接口
        int i,j;
        AnyType temp;
        for ( i = 1; i < arr.length; i++) {
            if(arr[i] != null){
                temp = arr[i];          //主要原理就在于compareTo的使用
                for ( j = i; j>0 && temp.compareTo(arr[j-1])< 0; j--) 
                    arr[j] = arr[j-1];
                arr[j] = temp;
            }
        }
        return arr;
    }*/
}  

Comparator的实现

  • 一下 Comparator 相关的代码 与 Comparable 接口不同之处在于实现 Compare 方法是在 Employee 类的外部,并且可以进行多种排序方式。
  • 主方法中调用排序函数也不相同 Array.sort(staff,new sortById)。
import java.util.Arrays;
import java.util.Comparator;
import java.util.Random;

public class CompartorStudy {

    public static void main(String[] args) {
        Employee1[] staff =  new Employee1[ 6 ];  
        staff[ 0 ] =  new Employee1( "harry Hacker" , 35000 );  
        staff[ 1 ] =  new Employee1( "carl cracke" , 25010 );  
        staff[ 2 ] =  new Employee1( "tony Tester" , 38200 );  
        staff[ 3 ] =  new Employee1( "Michale" , 23000 );
        staff[ 4 ] =  new Employee1( "Cindy" , 31100 );
        staff[ 5 ] =  new Employee1( "Yancy" , 32000 );
        
        Arrays.sort(staff,new sortById());
        for (Employee1 e : staff) {
            System.out.println( "id=" +e.getId()+ "  name=" +e.getName()+  
                     ".salary=" +e.getSalary());
        }
        Arrays.sort(staff,new sortBySalary());
        for (Employee1 e : staff) {
            System.out.println( "id=" +e.getId()+ "  name=" +e.getName()+  
                     ".salary=" +e.getSalary());
        }
    }

}

class Employee1{
    public Employee1(String name,double salary) {
        this.name = name;
        this.salary = salary;
        Random ID = new Random();
        id = ID.nextInt( 1000000 );
    }
    
    public int getId() {
        return id;
    }
    public void setId(int id) {
        this.id = id;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public double getSalary() {
        return salary;
    }
    public void setSalary(double salary) {
        this.salary = salary;
    }
    public void raiseSalary( double byPercent) {  
        double raise  = salary *byPercent/ 100 ;  
        salary+=raise;  
    } 
    private int id;
    private String name;
    private double salary;
}

class sortById implements Comparator<Employee1>{

    @Override
    public int compare(Employee1 o1, Employee1 o2) {
        return o1.getId() - o2.getId();
    }
}
class sortBySalary implements Comparator<Employee1>{

    @Override
    public int compare(Employee1 o1, Employee1 o2) {
        if (o1.getSalary() > o2.getSalary()) {
            return -1;
        }else if (o1.getSalary() < o2.getSalary()) {
            return 1;
        }else 
            return 0;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • 相同 Comparable和Comparator都是用来实现对象的比较、排序 要想对象比较、排序,都需要实现Com...
    Jeffbond阅读 6,236评论 3 20
  • 在收集对象之后,对象的排序是经常需要用到的操作。但我们不需要亲自实现各种排序算法,java.util.Collec...
    六尺帐篷阅读 375评论 0 1
  • java中我们要实现集合元素的比较和排序有两种方法。一种是让集合中的元素去实现Comparable接口,另一种是则...
    赵仝阅读 708评论 0 1
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,234评论 11 349
  • 男孩坐在窗边,看着外面的风景不断飞驰,想到再过几个小时就可以见到女孩,嘴角不由得上扬,浮现出一丝微笑。 男孩昨晚和...
    白水黑土阅读 240评论 0 1