相当于Java 7中的ComputeIfAbsent

有没有办法只在ConcurrentHashMap中不存在键时运行一段代码,并将代码的结果保存到集合中?

我不能使用Java 8功能,因为我正在为Android开发.

此外,我想避免运行长操作,如果我不需要,我不想打破集合的原子操作这样做.

解决方法

没有确切的等价物,但通常的方法是这样的:
ConcurrentMap<Key,Value> map = ...

Value computeIfAbsent(Key k) {
  Value v = map.get(k);
  if (v == null) {
    Value vNew = new Value(...); // or whatever else you do to compute the value
    v = (v = map.putIfAbsent(k,vNew)) == null ? vNew : v;
  }
  return v;
}

这在功能上与Java 8中的computeIfAbsent调用相当,唯一的区别在于有时你构造一个永远不会进入map的Value对象 – 因为另一个线程首先放入它.它永远不会导致返回错误的对象或类似的东西 – 无论如何,函数始终返回正确的值,但如果Value的构造具有副作用*,则这可能是不可接受的.

额外的实例通常不是性能问题,因为初始的get()检查消除了对putIfAbsent的大多数调用.通常,这种方法比computeIfAbsent快得多,因为当对象已经存在时,该调用会对对象进行不必要的锁定.如果有些物体严重争用,我在本地测量它的速度要快5倍.

如果您确实需要集成到映射中的计算行为(保持内部锁定以便一个线程创建新对象),则可以使用Guava的CacheBuilder获取LoadingCache.它与Java 8的CHM基本相同,但有大量额外的配置选项.

相关文章

最近看了一下学习资料,感觉进制转换其实还是挺有意思的,尤...
/*HashSet 基本操作 * --set:元素是无序的,存入和取出顺序不...
/*list 基本操作 * * List a=new List(); * 增 * a.add(inde...
/* * 内部类 * */ 1 class OutClass{ 2 //定义外部类的成员变...
集合的操作Iterator、Collection、Set和HashSet关系Iterator...
接口中常量的修饰关键字:public,static,final(常量)函数...