Java源码分析 - 策略模式在Java集合框架实现代码中的体现

发布于 2011-10-17 | 更新于 2020-09-20

在Java的集合框架中,构造Map或者Set时传入Comparator比较器,或者创建比较器传入Collections类的静态方法中作为方法的参数为Collection排序时,都使用到了策略模式

下面就以创建比较器传入Collections类的静态方法为例说明,下面是简单的调用代码:

LinkedList list = new LinkedList();
list.add(“arthinking”);
list.add(“Jason”);
list.add(“X”);
//创建一个逆序的比较器
Comparator r = Collections.reverseOrder();
//通过逆序的比较器进行排序
Collections.sort(list,r);

使用Collections.reverseOrder()方法床架一个比较器之后,再调用Collections.sort(list, r)把比较器传入该方法中进行排序,下面就看一下sort(list, r)中的代码:

public static void sort(List list, Comparator<? super T> c) {
Object[] a = list.toArray();
Arrays.sort(a, (Comparator)c);
ListIterator i = list.listIterator();
for (int j=0; j<a.length; j++) {
i.next();
i.set(a[j]);
}
}

Arrays.sort(a, (Comparator)c);这句继续把比较器传入处理,下面是Arrays.sort(a, (Comparator)c)的具体操作:

public static void sort(T[] a, Comparator<? super T> c) {
T[] aux = (T[])a.clone();
if (c==null)
mergeSort(aux, a, 0, a.length, 0);
else
mergeSort(aux, a, 0, a.length, 0, c);
}

这里又调用了mergeSort(aux, a, 0, a.length, 0, c);继续传递比较器,再跟进去,发现如下执行代码:

private static void mergeSort(Object[] src,
Object[] dest,
int low, int high, int off,
Comparator c) {
int length = high - low;

// Insertion sort on smallest arrays
if (length < INSERTIONSORT_THRESHOLD) {
    for (int i=low; i<high; i++)
	for (int j=i; j>low && c.compare(dest[j-1], dest[j])>0; j--)
	    swap(dest, j, j-1);
    return;
}

    // Recursively sort halves of dest into src
    int destLow  = low;
    int destHigh = high;
    low  += off;
    high += off;
    int mid = (low + high) >>> 1;
    mergeSort(dest, src, low, mid, -off, c);
    mergeSort(dest, src, mid, high, -off, c);

    // If list is already sorted, just copy from src to dest.  This is an
    // optimization that results in faster sorts for nearly ordered lists.
    if (c.compare(src[mid-1], src[mid]) <= 0) {
       System.arraycopy(src, low, dest, destLow, length);
       return;
    }

    // Merge sorted halves (now in src) into dest
    for(int i = destLow, p = low, q = mid; i < destHigh; i++) {
        if (q >= high || p < mid && c.compare(src[p], src[q]) <= 0)
            dest[i] = src[p++];
        else
            dest[i] = src[q++];
    }

}

这段代码有点复杂,我们跳出其中使用到比较器的代码分析:

c.compare(dest[j-1], dest[j])>0;

这里的compare方法在Comparator接口中也有定义:

public interface Comparator {
int compare(T o1, T o2);
}

由于这里使用泛型实现了Comparator,所以实际执行时,会根据比较器的具体实现类调用到实现代码,也就是上面创建的逆序比较器的compare方法,其实现如下:

public int compare(Comparable c1, Comparable c2) {
return c2.compareTo(c1);
}

从上面的跟踪代码执行流程,可以得出:

抽象策略角色类为:Comparator接口 真实策略角色类为:ReverseComparator 我们在这里传入不同的比较器就会得到不同的排序的执行方法了:

Collections.sort(list,r);

本文作者: arthinking

本文链接: https://www.itzhai.comjava-source-code-analysis-strategy-pattern-implementation-in-java-collections-framework-embodied-in-the-code.html

版权声明: 版权归作者所有,未经许可不得转载,侵权必究!联系作者请加公众号。

友情链接
破博客 IT宅
关联账号
公众号:IT宅
公众号:IT宅

IT宅技术博客公众号,博客更新动态同步至公众号。扫码关注公众号及时获取更新:↓↓↓

公众号:Java架构杂谈
公众号:Java架构杂谈

IT宅的Java专业技术公众号,只输出精品文章,更新评率较低。

联系我
1225538383@qq.com
arthinking
arthinking

微信联系我

更多信息
暂无

粤ICP备18081492号 © 2011 – 2025 IT宅. Designed by 帅旋. arthinking@itzhai. All Rights Reserved.
×
IT宅

关注公众号及时获取网站内容更新。

请帅旋喝一杯咖啡

咖啡=电量,给帅旋充杯咖啡,他会满电写代码!

IT宅

关注公众号及时获取网站内容更新。