【发布时间】:2012-10-17 15:34:30
【问题描述】:
在 Guava 中,是否有一种有效的方法可以向 ImmutableList 添加或删除项目(当然,在此过程中创建新列表)。
我能想到的最简单的方法是:
private ImmutableList<String> foos = ImmutableList.of();
public void addFoo(final String foo) {
if (this.foos.isEmpty()) {
foos = ImmutableList.of(foo);
} else {
foos = ImmutableList.<String>builder().addAll(foos).add(foo).build();
}
}
public void removeFoo(final String foo) {
final int index = this.foos.indexOf(foo);
if (index > -1) {
final Builder<String> builder = ImmutableList.<String>builder();
if (index > 0) builder.addAll(this.foos.subList(0, index));
final int size = this.foos.size();
if (index < size - 1) builder.addAll(this.foos.subList(index+1, size));
this.foos = builder.build();
}
}
我想避免这样做:
public void removeFoo(final String foo) {
final ArrayList<String> tmpList = Lists.newArrayList(this.foos);
if(tmpList.remove(foo))this.foos=ImmutableList.copyOf(tmpList);
}
但不幸的是,它比我能想到的任何纯 Guava 方法都简单得多。我错过了什么吗?
【问题讨论】:
-
你想解决什么更高层次的问题?如果您需要更改它们,也许您不应该处理不可变列表。
-
我知道当使用可变列表作为数据持有者之间时,我可以轻松解决问题,但是 a) 我必须创建多余的中间集合 b) 我必须将 java.util 集合与 guava ImmutableCollections 混合,而我想坚持一种范式。而且我想使用 ImmutableList 因为我想在
getFoos()方法中将其分发给客户端,而不会失去控制或不必创建大量 Collections.unmodifiableList 包装器对象。 -
然后我将使用可变列表作为实例字段并在 getFoos() 中返回 ImmutableList.copyOf(list)。
-
@ChristophLeiter 是的,但这意味着 a) 每次读取都需要完全遍历(我正在考虑读取比添加/删除调用更常见的场景),b) 我d 必须同步所有内容以避免 ConcurrentModificationExceptions。当然,我上面的示例也没有同步,但它们不会抛出 ConcurrentModificationExceptions。