当前位置: 首页 > 面试题库 >

CopyOnWriteArrayList和syncedList之间的区别

云昊阳
2023-03-14
问题内容

根据我的理解,并发集合类比同步集合更可取,因为并发集合类不会锁定完整的集合对象。相反,它们在收集对象的一小部分上锁定。

但是,当我检查的add方法时CopyOnWriteArrayList,我们正在获取完整集合对象的锁。那为什么CopyOnWriteArrayList比返回的列表更好Collections.synchronizedList呢?我在add方法中看到的唯一区别CopyOnWriteArrayList是,每次add调用该方法时,我们都会创建该数组的副本。

public boolean add(E e) {
    final ReentrantLock lock = this.lock;
    lock.lock();
    try {
        Object[] elements = getArray();
        int len = elements.length;
        Object[] newElements = Arrays.copyOf(elements, len + 1);
        newElements[len] = e;
        setArray(newElements);
        return true;
    } finally {
        lock.unlock();
    }
}

问题答案:

对于写入(添加)操作,CopyOnWriteArrayList使用ReentrantLock并创建数据的备份副本,并且仅通过setArray更新基础的易失性数组引用(在setArray之前的列表上的任何读取操作都将在添加之前返回旧数据)。
CopyOnWriteArrayList提供快照故障安全迭代器,并且在写入/添加操作时不会引发ConcurrentModifficationException。

但是,当我检查CopyOnWriteArrayList.class的add方法时,我们正在获取对完整集合对象的锁定。那么,CopyOnWriteArrayList为何比syncedList更好。我在CopyOnWriteArrayList的add方法中看到的唯一区别是,每次调用add方法时,我们都在创建该数组的副本。

  1. 不,锁不是在整个Collection对象上。如上所述,它是一个ReentrantLock,它与内部对象锁不同。
  2. add方法将始终创建现有数组的副本,并对副本进行修改,然后最终更新该数组的volatile引用以指向此新数组。这就是为什么我们使用“ CopyOnWriteArrayList”这个名称的原因-在您写入时创建副本。这也避免了ConcurrentModificationException


 类似资料:
  • 主要内容:文章目录,1.CopyOnWriteArrayList,2.CopyOnWriteArraySetCopyOnWrite的意思是,简单的说就是在对容器操作时,不是直接进行操作,而是先将容器复制一份,对新的容器操作后,再用新的容器覆盖原来的容器。 CopyOnWrite可以实现并发读操作,因为容器的写操作不会读当前容器造成影响。 CopyOnWrite的,不然n个写操作就会有n个容器副本,并且这n个副本会进行覆盖,最终只有1个副本的操作生效。 CopyOnWrite可以保证数据的,但是无

  • 问题内容: 我错放了太多次了,我想我一直忘记,因为我不知道两者之间的区别,只是一个给了我我期望的价值,而另一个却没有。 为什么是这样? 问题答案: 是的简写形式(尽管请注意,该表达式只会被计算一次。) 是的,即指定一元的到。 例子:

  • 问题内容: 因此,我有一段简单的代码可以打印出整数1-10: 然后,如果仅在第3行上更改一个运算符,它将打印出无限数量的1整数(我知道为什么会这样做)。为什么在运行第二个程序时没有出现语法错误?如果赋值运算符后面跟着一个加法运算符,它不会调用语法错误吗? 问题答案: 与相同, 只是意味着。

  • 问题内容: 有人可以解释一下 和 我不知道“确切”的含义 问题答案: 在这个例子中,什么都没有。当您具有多个具有相似名称的路径时,该参数将起作用: 例如,假设我们有一个显示用户列表的组件。我们还有一个用于创建用户的组件。的网址应嵌套在下。因此,我们的设置可能如下所示: 现在,这里的问题是,当我们转到路由器时,将通过所有定义的路由,并返回它找到的第一个匹配项。因此,在这种情况下,它将首先找到路线,然

  • 问题内容: 我很好奇printStackTrace()和toString()之间的区别是什么。乍一看,他们 似乎 做的完全相同。 码: 问题答案: 不,有重要区别!使用toString,您只有异常的类型和错误消息。使用printStackTrace()可以获得异常的整个堆栈跟踪,这对于调试非常有帮助。 System.out.println(toString())的示例: printStackTra