ConcurrentHashMap是否需要包装在同步块中? - java

是否需要将ConcurrentHashMap上的所有非还原操作(put()remove()等)包装在synchronized(this)块中?我知道所有这些操作都是线程安全的,因此这样做有真正的好处/需要吗?唯一使用的操作是put()remove()

protected final Map<String, String> mapDataStore = new ConcurrentHashMap<String, String>();

public void updateDataStore(final String key, final String value) {
    ...
    synchronized (this) {
        mapDataStore.put(key, value);
    }
    ...
}

参考方案

不,这样做会失去ConcurrentHashMap的好处。您也可以将HashMapsynchronizedsynchronizedMap()一起使用来锁定整个表(这是在synchronized中包装操作时要执行的操作,因为隐含的监视器是整个对象实例。)
ConcurrentHashMap的目的是通过允许在表上进行并发读/写而不锁定整个表来提高并发代码的吞吐量。该表通过使用锁条带(多个锁而不是一个)在内部支持此功能,每个锁分配给一组哈希桶-参见Goetz等人的Java Concurrency in Practice。

一旦使用了ConcurrentHashMap,由于实现中的锁条等,所有标准映射方法(put()remove()等)都成为原子的。唯一的权衡是,像size()isEmpty()这样的方法不一定会返回准确的结果,因为它们唯一的方法就是所有操作都可以锁定整个表。

ConcurrentMap interface接口还添加了新的原子复合操作,例如putIfAbsent()(仅当键尚未在地图中时才输入内容),remove()接受键和值(仅当其值等于您传递的参数时才删除条目)等。这些操作以前需要锁定整个表,因为它们需要两个方法调用才能完成(例如,如果您使用的是标准putIfAbsent()实现,则containsKey()将需要同时包装在put()块中的synchronizedMap的调用。)再一次,您可以获得使用这些方法可以避免锁定整个表,从而提高吞吐量。

Java:找到特定字符并获取子字符串 - java

我有一个字符串4.9.14_05_29_16_21,我只需要获取4.9。数字各不相同,所以我不能简单地获得此char数组的前三个元素。我必须找到最正确的.并将其子字符串化直到那里。我来自Python,因此我将展示Python的实现方法。def foobar(some_string): location = some_string.rfind('.&…

Java:线程池如何将线程映射到可运行对象 - java

试图绕过Java并发问题,并且很难理解线程池,线程以及它们正在执行的可运行“任务”之间的关系。如果我创建一个有10个线程的线程池,那么我是否必须将相同的任务传递给池中的每个线程,或者池化的线程实际上只是与任务无关的“工人无人机”可用于执行任何任务?无论哪种方式,Executor / ExecutorService如何将正确的任务分配给正确的线程? 参考方案 …

JAVA:字节码和二进制有什么区别? - java

java字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…

java:继承 - java

有哪些替代继承的方法? java大神给出的解决方案 有效的Java:偏重于继承而不是继承。 (这实际上也来自“四人帮”)。他提出的理由是,如果扩展类未明确设计为继承,则继承会引起很多不正常的副作用。例如,对super.someMethod()的任何调用都可以引导您通过未知代码的意外路径。取而代之的是,持有对本来应该扩展的类的引用,然后委托给它。这是与Eric…

Java:BigInteger,如何通过OutputStream编写它 - java

我想将BigInteger写入文件。做这个的最好方式是什么。当然,我想从输入流中读取(使用程序,而不是人工)。我必须使用ObjectOutputStream还是有更好的方法?目的是使用尽可能少的字节。谢谢马丁 参考方案 Java序列化(ObjectOutputStream / ObjectInputStream)是将对象序列化为八位字节序列的一种通用方法。但…