Spring中如何使用Comparator接口
我們先來回顧下Comparator接口在我們日常開發中的作用,Comparator比較器接口可以將自身傳遞給排序方法(比如Collections.sort或Arrays.sort),以便對排序順序進行精確控制。比如:
List<Integer> intList = Arrays.asList(2, 3, 1); Collections.sort(intList, (o1, o2) -> { return o2-o1; });
輸出
[3, 2, 1]
Comparator可以用來控制某些數據結構(如sored sets 或sorted maps)的順序,或者為沒有自然順序的對象集合提供順序,比如:
public class Emp { private int empNo; private String eName; //省略getter setter @Override public String toString() { return "empno:\t"+empno+"\tename:\t"+ename; } } Comparator<Emp> comparator = (o1, o2) -> { return o1.getEmpno()-o2.getEmpno(); }; empList.sort(comparator);
從以上使用方法可以看出Comparator是策略模式的一個經典體現,在Spring中它為實現瞭Ordered接口的類寫瞭一個比較器名叫OrderComparator,我們來看看它的compare方法:
@Override public int compare(@Nullable Object o1, @Nullable Object o2) { return doCompare(o1, o2, null); } private int doCompare(@Nullable Object o1, @Nullable Object o2, @Nullable OrderSourceProvider sourceProvider) { //判斷是否實現瞭PriorityOrdered接口 //o1為原先的後一個元素 //o2為原先的前一個元素 boolean p1 = (o1 instanceof PriorityOrdered); boolean p2 = (o2 instanceof PriorityOrdered); //p1實現瞭PriorityOrdered,但是p2沒實現PriorityOrdered,則p1優先級更高 if (p1 && !p2) { ////小於0 表示逆序 o1排前 return -1; } else if (p2 && !p1) { //p2實現瞭PriorityOrdered,但是p1沒實現PriorityOrdered,則p2優先級更高 //大於0 表示正序 o2排前 return 1; } int i1 = getOrder(o1, sourceProvider); int i2 = getOrder(o2, sourceProvider); return Integer.compare(i1, i2); } protected int getOrder(@Nullable Object obj) { if (obj != null) { Integer order = findOrder(obj); if (order != null) { return order; } } //Integer.MAX_VALUE代表最低優先級 return Ordered.LOWEST_PRECEDENCE; } @Nullable protected Integer findOrder(Object obj) { return (obj instanceof Ordered ? ((Ordered) obj).getOrder() : null); }
OrderComparator主要是通過實現Order接口方法getOrder返回的值來比較的,而通過註解Order比較的話是采用繼承OrderComparator類的AnnotationAwareOrderComparator類來重載getOrder方法來實現:
@Override @Nullable protected Integer findOrder(Object obj) { // Check for regular Ordered interface //檢查常規的Order接口 Integer order = super.findOrder(obj); if (order != null) { return order; } //檢查@Order註解和@Priority註解 // Check for @Order and @Priority on various kinds of elements if (obj instanceof Class) { return OrderUtils.getOrder((Class<?>) obj); } else if (obj instanceof Method) { Order ann = AnnotationUtils.findAnnotation((Method) obj, Order.class); if (ann != null) { return ann.value(); } } else if (obj instanceof AnnotatedElement) { //AnnotatedElement代表在當前運行的java虛擬機中一個可以被註解的元素,這個接口允許通過反射讀取元素上面的註解,這與我們之 前的理解是一致的 //https://www.yuque.com/cuihualong/javaseries/sh91es Order ann = AnnotationUtils.getAnnotation((AnnotatedElement) obj, Order.class); if (ann != null) { return ann.value(); } } else { order = OrderUtils.getOrder(obj.getClass()); if (order == null && obj instanceof DecoratingProxy) { order = OrderUtils.getOrder(((DecoratingProxy) obj).getDecoratedClass()); } } return order; }
這裡順便提下在SpringBoot中ConditionalOnBean,ConditionalOnProperty,ConditionalOnClass等註解實際上都采用瞭實現Order註解的方法來進行排序。這裡我就截個圖看下瞭:
Conditional註解中的OnClassCondition類就標瞭@Order註解
實現Comparator接口compare方法的時候主要是要註意以下幾點(翻譯自官方文檔):
用中文簡單來說就是在 JDK7 版本以上默認使用Timsort排序方法來實現,Comparator 比較器要滿足自反性,傳遞性,反對稱性,不然 Arrays.sort,Collections.sort有可能會報 IllegalArgumentException 異常。
以上就是Spring中如何使用Comparator接口的詳細內容,更多關於Spring 使用Comparator接口的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- Spring @Order註解使用詳解
- Spring筆記-@Order註解和Ordered接口解析
- 深入淺出講解Java比較器及數學常用類
- Java Comparator.comparing比較導致空指針異常的解決
- 基於Arrays.sort()和lambda表達式