RandomAccess 明明是个空接口,能有什么用呢?

简介: Hello,大家好,我是阿粉,Java 语言中有很多有意思的设计,之前二哥的一篇文章中介绍了Serializable 空接口,今天阿粉给大家介绍一个另一个 Java 中的空接口 RandomAccess。

背景

打开 JDK 源码,搜索 RandomAccess 我们可以看到,如下代码块,就是一个申明,里面空空如也,啥也没有。要是放在以前阿粉肯定会觉得这种写法肯定没什么用,但是毕竟看过二哥之前的文章,阿粉已经学到了,不会再那么年轻了。

public interface RandomAccess {
}

在介绍RandomAccess 的作用之前我们先看下集合 List 的两个基本的实现ArrayListLinkedList ,这两个的区别主要有下面几个

  1. 底层数据结构不同,Arraylist 底层使用的是 Object 数组;LinkedList 底层使用的是双向链表数据结构;
  2. 添加删除的时间复杂度不同,上面提到因为ArrayList 底层采用的是数组结构存储数据的,所以在插入和删除元素的时候,时间复杂度受元素位置的影响。在执行添加方法的时候, ArrayList 会默认将元素添加到数组的最后,这种情况时间复杂度就是 O(1)。但是如果要在指定位置 i 插入和删除元素的话时间复杂度就为 O(n-i) ,因为底层的数组需要进行移动操作。而 LinkedList 底层采用链表存储,对于添加元素和删除元素的时间复杂度就不会受元素位置的影响,近似 O(1),如果是要在指定位置 i 插入和删除元素的话时间复杂度近似为 o(n)) 因为需要先遍历移动到指定位置再插入。
  3. 快速随机访问机制不同,LinkedList 不支持高效的随机元素访问,而 ArrayList 支持,快速随机访问就是通过元素的序号快速获取元素对象(对应于get(int index)方法

揭秘

正是由于 ArrayListLinkedList 的随机访问性不同,才有了RandomAccess 空接口的存在!

看下面代码,我们创建 ArrayListLinkedList 两个集合,然后采用二分查找,找到指定的内容。我们顺藤摸瓜,可以发现RandomAccess 接口在Collections 类中binarySearch 方法中有使用,而且在给定的集合签名不同的时候会执行不同的查询方式,源码如下:

public class ListTest {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("string1");
        list.add("string2");
        list.add("string3");
        int string2 = Collections.binarySearch(list, "string2");
        System.out.println(string2);
        LinkedList<String> linkedList = new LinkedList<>();
        linkedList.add("string4");
        linkedList.add("string5");
        linkedList.add("string6");
        int string5 = Collections.binarySearch(linkedList, "string6");
        System.out.println(string5);
    }
}
 public static <T> int binarySearch(List<? extends Comparable<? super T>> list, T key) {
    if (list instanceof RandomAccess || list.size()<BINARYSEARCH_THRESHOLD)
        return Collections.indexedBinarySearch(list, key);
    else
        return Collections.iteratorBinarySearch(list, key);
}

可以明显的看到在这个二分查找的方法里面根据 List 是否实现了 RandomAccess 接口来决定采用哪个搜索方案,如果实现了RandomAccess 接口表示这个List 支持随机访问,所以采用的是常规的二分查找;如果没有实现RandomAccess接口,则采用的迭代器的二分查找,因为迭代器的二分查找的实现效率相对较低,源码如下:

private static <T>
    int indexedBinarySearch(List<? extends Comparable<? super T>> list, T key) {
        int low = 0;
        int high = list.size()-1;
        while (low <= high) {
            int mid = (low + high) >>> 1;
            Comparable<? super T> midVal = list.get(mid);
            int cmp = midVal.compareTo(key);
            if (cmp < 0)
                low = mid + 1;
            else if (cmp > 0)
                high = mid - 1;
            else
                return mid; // key found
        }
        return -(low + 1);  // key not found
    }
    private static <T>
    int iteratorBinarySearch(List<? extends Comparable<? super T>> list, T key)
    {
        int low = 0;
        int high = list.size()-1;
        ListIterator<? extends Comparable<? super T>> i = list.listIterator();
        while (low <= high) {
            int mid = (low + high) >>> 1;
            Comparable<? super T> midVal = get(i, mid);
            int cmp = midVal.compareTo(key);
            if (cmp < 0)
                low = mid + 1;
            else if (cmp > 0)
                high = mid - 1;
            else
                return mid; // key found
        }
        return -(low + 1);  // key not found
    }
    private static <T> T get(ListIterator<? extends T> i, int index) {
        T obj = null;
        int pos = i.nextIndex();
        if (pos <= index) {
            do {
                obj = i.next();
            } while (pos++ < index);
        } else {
            do {
                obj = i.previous();
            } while (--pos > index);
        }
        return obj;
    }

从上面的源码我们可以看到,都是采用二分查找,但是不同的地方是,indexedBinarySearch 方法中使用的是list.get(index)) 来获取中值的,而在iteratorBinarySearch 方法中是使用get(i, mid); 来获取中值的,get(i, mid); 中是通过while 遍历来查找中值的,因此效率会相对较低。

总结

通过上面的代码我们发现,其实RandomAccess 这个空接口就是用来作为标识的,标识某个集合是否支持随机访问,支持随机访问的话在进行搜索的时候我们可以采用更高效的搜索方案。

相关文章
|
7月前
|
JSON Java 测试技术
为了理直气壮怼回去,写了一个日志切面输出接口出入参数
我们在日常排查问题过程中知道,入参传错是导致接口调用失败的常见原因之一。特别是提供给第三方调用的**回调接口和openAPI接口**,由于无法保证第三方开发人员的水平,经常问题不断,反反复复找你问为啥掉不通,甚至吐槽写的“啥玩意接口”,这时候你肯定一脸懵逼,怒火中烧,想展开撕逼甩锅大战,但是对方有可能是甲方金主爸爸并且你没有第一时间掌握证据证明证是对方调用的问题,你只能忍着问他是如何调接口的,卑微请求他把传参发过来看看。。。为了扭转局势,挺直腰杆怼回去:能不能靠谱点?今天我们就来讲讲系统服务中如何优雅地实现统一打印接口API参数日志,方便服务端开发快速甩锅还能拿出证据!!!
93 0
为了理直气壮怼回去,写了一个日志切面输出接口出入参数
|
机器学习/深度学习 算法 C++
没什么了。
没什么了。
|
算法 C++
没什么。。。。
没什么。。。。
|
前端开发 计算机视觉 Python
代码报错还好说,源码报错才难搞!分享自己源码报错的解决过程!
代码报错还好说,源码报错才难搞!分享自己源码报错的解决过程!
109 0
代码报错还好说,源码报错才难搞!分享自己源码报错的解决过程!
|
JSON Java fastjson
Springboot 最细节全面的接口传参接参介绍,总有你喜欢的一种方式
Springboot 最细节全面的接口传参接参介绍,总有你喜欢的一种方式
917 0
Springboot 最细节全面的接口传参接参介绍,总有你喜欢的一种方式
|
JavaScript 前端开发
如何确保你的构造函数只能被new调用,而不能被普通调用?| 踩坑日记
如何确保你的构造函数只能被new调用,而不能被普通调用?| 踩坑日记
558 0
如何确保你的构造函数只能被new调用,而不能被普通调用?| 踩坑日记
|
JSON 数据可视化 JavaScript
开发改了接口,经常忘通知测试,有什么好的解决方案吗?
不知道大家有没有同感,做接口测试麻烦的不是测试本身,而是接口它会变,更麻烦的不是接口变了,而是它变了而你不知道。等到你测完,开发才悠悠跟你说——“那个接口我改了点东西,你再看一眼哈”。
开发改了接口,经常忘通知测试,有什么好的解决方案吗?
答网友提问,ABAP post调用外部接口遇到405 not allow
答网友提问,ABAP post调用外部接口遇到405 not allow
132 0
答网友提问,ABAP post调用外部接口遇到405 not allow
|
人工智能 达摩院 前端开发
想爱的人,想去的地方,一秒钟都不想等了!
万物复苏,终于复工。上周,橙子在内网上向阿里同学发起提问:2020我们拥抱变变变化,但无论发生什么2020你都想「做成」的事是什么?
255 0
想爱的人,想去的地方,一秒钟都不想等了!
|
API 调度 C语言
视频图像处理中的错帧同步是怎么实现的?
错帧同步,简单来说就是把当前的几帧缓冲到子线程中处理,主线程直接返回子线程之前的处理结果,属于典型的以空间换时间策略。
349 0
视频图像处理中的错帧同步是怎么实现的?