【发布时间】:2011-05-31 17:43:37
【问题描述】:
我有一个 java 类,它可以同时被很多线程访问,并希望确保它是线程安全的。该类有一个私有字段,即字符串到字符串列表的映射。我已将 Map 实现为 ConcurrentHashMap 以确保获取和放置是线程安全的:
public class ListStore {
private Map<String, List<String>> innerListStore;
public ListStore() {
innerListStore = new ConcurrentHashMap<String, List<String>>();
}
...
}
因此,鉴于 Map 的获取和放置是线程安全的,我关心的是存储在 Map 中的列表。例如,考虑以下方法来检查给定条目是否存在于商店的给定列表中(为简洁起见,我省略了错误检查):
public boolean listEntryExists(String listName, String listEntry) {
List<String> listToSearch = innerListStore.get(listName);
for (String entryName : listToSearch) {
if(entryName.equals(listEntry)) {
return true;
}
}
return false;
}
看来我需要同步该方法的全部内容,因为如果另一个方法在此方法迭代时更改了 innerListStore.get(listName) 处的列表内容,则会引发 ConcurrentModificationException。
正确吗?如果正确,我是在 innerListStore 上同步还是在本地 listToSearch 变量上同步?
更新:感谢您的回复。听起来我可以在列表本身上同步。更多信息,这里是 add() 方法,它可以在 listEntryExists() 方法在另一个线程中运行的同时运行:
public void add(String listName, String entryName) {
List<String> addTo = innerListStore.get(listName);
if (addTo == null) {
addTo = Collections.synchronizedList(new ArrayList<String>());
List<String> added = innerListStore.putIfAbsent(listName, addTo);
if (added != null) {
addTo = added;
}
}
addTo.add(entryName);
}
如果这是修改存储在映射中的基础列表的唯一方法并且没有公共方法返回对映射的引用或映射中的条目,我可以在列表本身上同步迭代吗? add() 的这种实现是否足够?
【问题讨论】:
-
您的
add()实现已损坏。你需要正确处理putIfAbsent()的结果(否则你可能会添加到错误的列表中)。 -
@jtahlborn 你是说我需要在 putIfAbsent() 返回的 List 上调用 add() 吗?如果是这样,我不同意。 putIfAbsent() 返回与该键关联的以前的任何内容,这将是错误的列表。对吗?
-
请重新阅读有关
putIfAbsent()方法的文档(请注意该方法的名称)。 -
我想我现在明白你的意思并编辑了 add()。这是正确的处理方式吗?
标签: java concurrency