如何对CopyOnWriteArrayList进行排序

Jok*_*ker 6 java core copyonwritearraylist

我想对CopyOnWriteArrayList进行排序.目前它正在抛出未排序的操作异常.

  import java.util.Collections;
  import java.util.List;
  import java.util.concurrent.CopyOnWriteArrayList;

 public class CopyOnWriteArrayListExample {

  public static void main(final String[] args) {
     List<String> list = new CopyOnWriteArrayList<>();
    list.add("3");
    list.add("2");
    list.add("1");

    Collections.sort(list);
   }
}

Exception in thread "main" java.lang.UnsupportedOperationException
at java.util.concurrent.CopyOnWriteArrayList$COWIterator.set(CopyOnWriteArrayList.java:1049)
at java.util.Collections.sort(Collections.java:159)
at com.sac.list.CopyOnWriteArrayListExample.main(CopyOnWriteArrayListExample.java:15)  
Run Code Online (Sandbox Code Playgroud)

提前致谢.

Evg*_*eev 9

Collections.sort使用ListIterator.set

    ...
    for (int j=0; j<a.length; j++) {
        i.next();
        i.set((T)a[j]);
    }
Run Code Online (Sandbox Code Playgroud)

但CopyOnWriteArrayList的ListIterator不支持删除,设置或添加方法.

解决方法:

    Object[] a = list.toArray();
    Arrays.sort(a);
    for (int i = 0; i < a.length; i++) {
        list.set(i, (String) a[i]);
    }
Run Code Online (Sandbox Code Playgroud)


Mar*_*bst 5

叶夫根尼(Evgeniy)的解决方案指向正确的方向,但是list.set(i, (String) a[i])必须锁定list列表中的每个元素。如果有写入的并发线程list将大大减慢循环。

为了最大程度地减少阻塞,最好减少更改的语句数list

    CopyOnWriteArrayList<Integer> list = new CopyOnWriteArrayList<>();

    // ... fill list with values ...

    ArrayList<Integer> temp = new ArrayList<>();
    temp.addAll(list);                           
    Collections.sort(temp);

    list.clear();            // 1st time list is locked
    list.addAll(temp);       // 2nd time list is locked
Run Code Online (Sandbox Code Playgroud)

不利之处在于,如果并发线程list在两者之间进行读取,clear()并且addAll(temp)使用Evgeniy的解决方案,它将看到一个空列表,那么它可能会看到部分排序的列表。


小智 5

在JDK1.8中可以sort(Comparator<? super E> c)直接使用.

List<Integer> list = new CopyOnWriteArrayList<Integer>();

list.add(3);
list.add(4);
list.add(1);

list.sort(new Comparator<Integer>() {
    @Override
    public int compare(Integer o1, Integer o2) {
        return o1 - o2;
    }
});
Run Code Online (Sandbox Code Playgroud)

  • 这需要minSDK 24 (2认同)