1. 不安全的Map
示例代码:
代码语言:javascript复制public class ListDemo {
public static void main(String[] args) throws InterruptedException {
// 定义一个存储
Map map = new HashMap();
// 循环生成10个线程
for (int i = 0; i < 30; i ) {
// 每个线程存入10000个数据
new Thread(() -> {
for (int j = 0; j < 1000; j ) {
map.put(Thread.currentThread().getName() j, UUID.randomUUID());
}
}).start();
}
// 休眠等待线程跑完
Thread.sleep(100);
// 输出集合数量
System.out.println("map的数量为:" map.size());
}
}
示例结果:
2. 线程安全的解决方法
- Collections
使用 Collections.synchronizedMap(new HashMap<>()); 将HashMap转换为安全的Map
但是无论是读取还是写入,它都会进行加锁,当我们并发级别特别高,线程之间在任何操作上都会进行等待,因此在某些场景中它不是最好的选择。
- ConcurrentHashMap
ConcurrentHashMap 采用了分段锁技术,其中 Segment 继承于 ReentrantLock。不会像 HashTable 那样不管是 put 还是 get 操作都需要做同步处理,理论上 ConcurrentHashMap 支持 CurrencyLevel (Segment 数组数量)的线程并发。每当一个线程占用锁访问一个 Segment 时,不会影响到其他的 Segment。