集册 Java 并发编程 多线程环境下安全使用集合 API

多线程环境下安全使用集合 API

欢马劈雪     最近更新时间:2020-08-04 05:37:59

178

在集合 API 中,最初设计的 Vector 和 Hashtable 是多线程安全的。例如:对于 Vector 来说,用来添加和删除元素的方法是同步的。如果只有一个线程与 Vector 的实例交互,那么,要求获取和释放对象锁便是一种浪费,另外在不必要的时候如果滥用同步化,也有可能会带来死锁。因此,对于更改集合内容的方法,没有一个是同步化的。集合本质上是非多线程安全的,当多个线程与集合交互时,为了使它多线程安全,必须采取额外的措施。

在 Collections 类中有多个静态方法,它们可以获取通过同步方法封装非同步集合而得到的集合:

  • public static Collection synchronizedCollention(Collection c)

  • public static List synchronizedList(list l)

  • public static Map synchronizedMap(Map m)

  • public static Set synchronizedSet(Set s)

  • public static SortedMap synchronizedSortedMap(SortedMap sm)

  • public static SortedSet synchronizedSortedSet(SortedSet ss)

这些方法基本上返回具有同步集合方法版本的新类。比如,为了创建多线程安全且由 ArrayList 支持的 List,可以使用如下代码:

List list = Collection.synchronizedList(new ArrayList());

注意,ArrayList 实例马上封装起来,不存在对未同步化 ArrayList 的直接引用(即直接封装匿名实例)。这是一种最安全的途径。如果另一个线程要直接引用 ArrayList 实例,它可以执行非同步修改。

下面给出一段多线程中安全遍历集合元素的示例。我们使用 Iterator 逐个扫描 List 中的元素,在多线程环境中,当遍历当前集合中的元素时,一般希望阻止其他线程添加或删除元素。安全遍历的实现方法如下:

展开阅读全文