Collections是一个工具类,sort是其中的静态方法,是用来对List类型进行排序的,它有两种参数形式:


public static <T extends Comparable<? super T>> void sort(List<T> list) {
        list.sort(null);
    }
public static <T> void sort(List<T> list, Comparator<? super T> c) {
        list.sort(c);
    }

首先使用基本类型(此处使用Integer)来演示第一个方法:

static Integer[] a = {2,3,1};	
 static List<Integer> list=new ArrayList<Integer>(Arrays.asList(a));	
	public static void main(String[] args) {
		System.out.println("排序前:"+list);
	        System.out.println("读书百遍,其义自见");
	        Collections.sort(list);
	        System.out.println("排序后:"+list);
	}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_逆序

 

可以看到,默认的排序是正序,那么如何实现逆序呢,这就要使用第二种方式了,即通过实现Comparator接口的compare方法来完成自定义排序,代码如下:

static Integer[] a = { 2, 3, 1 };
	static List<Integer> list = new ArrayList<Integer>(Arrays.asList(a));
	public static void main(String[] args) {
		System.out.println("排序前:" + list);
		System.out.println("读书百遍,其义自见");
		Collections.sort(list, new Comparator<Integer>() {
			public int compare(Integer o1, Integer o2) {
				// 返回值为int类型,大于0表示正序,小于0表示逆序
				return o2 - o1;
			}
		});
		System.out.println("排序后:" + list);
	}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_System_02

可以看到,已经实现了逆序的排序了。

接下来看看自定义类的排序:

定义一个Student类:

public class Student {

	    private int sno; //学号
	    private String name;//姓名
	    
		public int getSno() {
			return sno;
		}
		public void setSno(int sno) {
			this.sno = sno;
		}
		public String getName() {
			return name;
		}
		public void setName(String name) {
			this.name = name;
		}
		
		public Student(int sno, String name) {
			super();
			this.sno = sno;
			this.name = name;
		}
		
		@Override
		public String toString() {
			return "Student [sno=" + sno + ", name=" + name + "]";
		}   		
}

首先使用同样的方式来使用Collections.sort方法:


java collections.sort java collections.sort排序 原理_List_03


此时会报错:

Bound mismatch: The generic method sort(List<T>) of type Collections is not applicable for the arguments (List<Student>). The inferred type Student is not a valid substitute for the bounded parameter <T extends Comparable<? super T>>

意思是参数类型为List<Student>时,sort方法无法执行,原因是泛型没有继承Comparable接口,这种方式稍后再说,我们先使用sort方法的第二种形式:

根据不同的排序方式即可呈现不同的结果。

1.按照学生学号正序排序

public static void main(String[] args) {	
			 Student stu1 = new Student(2,"至尊宝");
			 Student stu2 = new Student(3,"凤求凰");
			 Student stu3 = new Student(1,"永耀之星");
			 List<Student> list = new ArrayList<Student>(Arrays.asList(stu1,stu2,stu3));
			 System.out.println("排序前:"+list);
			 System.out.println("读书百遍,其义自见");
			 Collections.sort(list,new Comparator<Student>() {
				public int compare(Student o1, Student o2) {
					//按学号正序排序
					return o1.getSno()-o2.getSno();
				}
			});
			 System.out.println("排序后:"+list);
		}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_List_04

2.按照学生学号逆序排序

public static void main(String[] args) {	
			 Student stu1 = new Student(2,"至尊宝");
			 Student stu2 = new Student(3,"凤求凰");
			 Student stu3 = new Student(1,"永耀之星");
			 List<Student> list = new ArrayList<Student>(Arrays.asList(stu1,stu2,stu3));
			 System.out.println("排序前:"+list);
			 System.out.println("读书百遍,其义自见");
			 Collections.sort(list,new Comparator<Student>() {
				public int compare(Student o1, Student o2) {
					//按学号逆序排序
					return o2.getSno()-o1.getSno();
				}
			});
			 System.out.println("排序后:"+list);
		}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_逆序_05

3.按照学校姓名正序排序

public static void main(String[] args) {	
			 Student stu1 = new Student(2,"至尊宝");
			 Student stu2 = new Student(3,"凤求凰");
			 Student stu3 = new Student(1,"永耀之星");
			 List<Student> list = new ArrayList<Student>(Arrays.asList(stu1,stu2,stu3));
			 System.out.println("排序前:"+list);
			 System.out.println("读书百遍,其义自见");
			 Collections.sort(list,new Comparator<Student>() {
				public int compare(Student o1, Student o2) {
					//按学生姓名正序排序
					return o1.getName().compareTo(o2.getName());
				}
			});
			 System.out.println("排序后:"+list);
		}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_List_06


4.按照学生姓名逆序排序:

public static void main(String[] args) {	
			 Student stu1 = new Student(2,"至尊宝");
			 Student stu2 = new Student(3,"凤求凰");
			 Student stu3 = new Student(1,"永耀之星");
			 List<Student> list = new ArrayList<Student>(Arrays.asList(stu1,stu2,stu3));
			 System.out.println("排序前:"+list);
			 System.out.println("读书百遍,其义自见");
			 Collections.sort(list,new Comparator<Student>() {
				public int compare(Student o1, Student o2) {
					//按学生姓名逆序排序
					return o2.getName().compareTo(o1.getName());
				}
			});
			 System.out.println("排序后:"+list);
		}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_System_07


接下来看看第一种形式的实现

首先让Student类继承Comparable接口并重写compareTo方法(为了和上面的排序方式区别开,此次按照学生姓名逆序排列):


public class Student implements Comparable<Student>{
	
    private int sno;
    private String name;
    
	public int getSno() {
		return sno;
	}
	public void setSno(int sno) {
		this.sno = sno;
	}
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}
	private Student(int sno, String name) {
		super();
		this.sno = sno;
		this.name = name;
	}
	@Override
	public String toString() {
		return "Student [name=" + name + ", sno=" + sno + "]";
	}
	public int compareTo(Student stu) {
		//按学号正序排序
		//return this.getSno()-stu.getSno();
		//按学号逆序排序
		//return stu.getSno()-this.getSno();
		//按学生姓名正序排序
		//return this.getName().compareTo(stu.getName());
		//按学生姓名逆序排序
		return stu.getName().compareTo(this.getName());
	}
	
	public static void main(String[] args) {
		 Student stu1 = new Student(2,"至尊宝");  
	     Student stu2 = new Student(3,"凤求凰");  
	     Student stu3 = new Student(1,"永耀之星");  
	     List<Student> list = new ArrayList<Student>(Arrays.asList(stu1,stu2,stu3));  
		 System.out.println("排序前:"+list);  
	     System.out.println("读书百遍,其义自见");  
	     Collections.sort(list);  
	     System.out.println("排序后:"+list);  
	}   
}

运行结果如下图所示:

java collections.sort java collections.sort排序 原理_逆序_08



总结:

1.对于String或Integer这些已经实现Comparable接口的类来说,可以直接使用Collections.sort方法传入list参数来实现默认方式(正序)排序;

如果不想使用默认方式(正序)排序,可以通过Collections.sort传入第二个参数类型为Comparator来自定义排序规则;

3.对于自定义类型(如本例子中的Student),如果想使用Collections.sort的方式一进行排序,可以通过实现Comparable接口的compareTo方法来进行,如果不实现,则参考第2点;

jdk1.8的Comparator接口有很多新增方法,其中有个reversed()方法比较实用,是用来切换正序和逆序的,代码如下:

System.out.println("排序前:"+list);  
	     System.out.println("读书百遍,其义自见");  
	     Comparator<Student> comparator =new Comparator<Student>() {
			public int compare(Student o1, Student o2) {
				//按学号正序排序
				//return o1.getSno()-o2.getSno();
				//按学号逆序排序
				//return o2.getSno()-o1.getSno();
				//按学生姓名正序排序
				//return o1.getName().compareTo(o2.getName());
				//按学生姓名逆序排序
				return o2.getName().compareTo(o1.getName());
			}
		};
		/*新的排序实现方式*/
	     Collections.sort(list,comparator.reversed());  
	     System.out.println("排序后:"+list);

运行结果如下图所示:

重写的compare方法定义的是按学生姓名逆序排序,在使用reversed翻转后是按学生姓名正序排序,结果如下:

java collections.sort java collections.sort排序 原理_System_09