CopyOnWriteArrayList(免锁容器)的好处之一長当多个迭代器同时遍历和修改这个列表时,不 会抛出 ConcurrentModificationException。在 CopyOnWriteArrayList 中,写入将导致创建 整个底层数组的副本,而源数组将保留在原地,使得复制的数组在被修改时,读取操作可以安 全地执行。
1、 由于写操作的时候,需要拷贝数组,会消耗内存,如果原数组的内容比较多的情况下,可 能导致young gc或者full gc;
2、 不能用于实时读的场景,像拷贝数组、新增元素都需要时间,所以调用一个set操作后, 读取到数据可能还是旧的,虽然CopyOnWriteArrayList能做到最终一致性,但是还是没法 满足实时性要求;
CopyOnWriteArrayList 透露的思想
1、 读写分离,读和写分开
2、 最终一致性
3、 使用另外开辟空间的思路,来解决并发冲突
3、 就绪态该状态下的线程已经获得执行所需的所有资源,只要CPU分配执行权就能运行。 所有就绪态的线程存放在就绪队列中。
4、 运行态获得CPU执行权,正在执行的线程。由于一个CPU同一时刻只能执行一条线程, 因此每个CPU每个时刻只有一条运行态的线程。
阻塞态:
当一条正在执行的线程请求某一资源失败时,就会进入阻塞态。而在Java中,阻塞态专指请 求锁失败时进入的状态。由一个阻塞队列存放所有阻塞态的线程。处于阻塞态的线程会不断请 求资源,一旦请求成功,就会进入就绪队列,等待执行。PS:锁、I0、Socket等都资源。
等待态:
当前线程中调用wait、join、park函数时,当前线程就会进入等待态。也有一个等待队列存 放所有等待态的线程。线程处于等待态表示它需要等待其他线程的指示才能继续运行。进入等 待态的线程会释放CPU执行权,并释放资源(如:锁)
超时等待态:
当运行中的线程调用sleep (time), wait, join、parkNanos, parkUntil时,就会进入该状态; 它和等待态一样,并不是因为请求不到资源,而是主动进入,并且进入后需要其他线程唤醒; 进入该状态后释放CPU执行权和占有的资源。与等待态的区别:到了超时时间后自动进入阻 塞队列,开始竞争锁。
终止态:
线程执行结束后的状态。
注意:
1、 wait()方法会释放CPU执行权和占有的锁。
2、 sleep (long)方法仅释放CPU使用权,锁仍然占用;线程被放入超时等待队列,与yield 相比,它会使线程较长时间得不到运行。
3、 yield()方法仅释放CPU执行权,锁仍然占用,线程会被放入就绪队列,会在短时间内再 次执行。
4、 wait和notify必须配套使用,即必须使用同一把锁调用;
Was this helpful?
1 / 0