多线程中对某些共享变量进行读写操作,并发读操作时不会产生问题,并发读写或并发写操作会引起线程安全性问题。
下面代码会报错:java.util.ConcurrentModificationException
public class TestReentrantReadWriteLock { volatile Map dataMap = new HashMap(); private static TestReentrantReadWriteLock testReentraintReadWriteLock = new TestReentrantReadWriteLock(); private static TestReentrantReadWriteLock getInstace(){ return testReentraintReadWriteLock; } public void readMap(){ Map refMap = dataMap; Iterator iterator = refMap.keySet().iterator(); while(iterator.hasNext()){//这行报错:java.util.ConcurrentModificationException Object key = iterator.next(); Object value = refMap.get(key); } } public void writeMap(){ dataMap.clear(); for(int i=0;i<100;i++){ dataMap.put(i+"", i+""); } } public static void main(String args []){ for(int i=0;i<2;i++){ new Thread(new Runnable(){ @Override public void run() { // TODO Auto-generated method stub\ while(true){ TestReentrantReadWriteLock.getInstace().readMap(); } } }).start(); } for(int i=0;i<1;i++){ new Thread(new Runnable(){ @Override public void run() { while(true){ TestReentrantReadWriteLock.getInstace().writeMap(); } } }).start(); } } }
解决方法:
1 如果只有一条线程会执行写操作, 可以在写操作时新建一个MAP(业务逻辑上可行的话),然后替换原来的MAP:
public void writeMap(){ Map newMap = new HashMap(); for(int i=0;i<100;i++){ newMap.put(i+"", i+""); } dataMap = newMap; }
1.使用java.util.concurrent.ConcurrentHashMap 代替HashMap 解决问题。
2.使用synchronized 锁,需要加到 writeMap 和 readMap方法上,可以决问题,但多个读操作互斥,影响并发。
3.读操作远大于写操作情况下可以使用ReentrantReadWriteLock
示例代码:
ReentrantReadWriteLock rwl = new ReentrantReadWriteLock(); public void readMap(){ rwl.readLock().lock();//读取锁 读取与读取可并发,与写入互斥 try{ Map refMap = dataMap; Iterator iterator = refMap.keySet().iterator(); while(iterator.hasNext()){ Object key = iterator.next(); Object value = refMap.get(key); } }finally{ rwl.readLock().unlock();//解锁 } } public void writeMap(){ rwl.writeLock().lock();//写入锁 写入与读取之间互斥 写入与写入之间互斥 try{ dataMap.clear(); for(int i=0;i<100;i++){ dataMap.put(i+"", i+"");//解锁 } }finally{ rwl.writeLock().unlock(); } }
相关推荐
读写锁ReentrantReadWriteLock&StampLock详解_e读写锁ReentrantReadWriteLock&StampLock详解_e读写锁ReentrantReadWriteLock&StampLock详解_e读写锁ReentrantReadWriteLock&StampLock详解_e读写锁...
【深入浅出ReentrantReadWriteLock源码解析】 ReentrantReadWriteLock是Java并发包中的一个核心类,它提供了读写锁的实现,使得多个线程可以并发读取共享资源,但写操作是互斥的,即同一时间只能有一个线程进行写...
根据提供的文件信息,本文将详细解析读写锁`ReentrantReadWriteLock`以及`StampLock`在Java并发编程中的应用场景及其实现原理。 ### 一、读写锁介绍 #### 1.1 读写锁的基本概念 读写锁是一种特殊的锁机制,它可以...
在Java多线程并发编程中,ReentrantReadWriteLock(可重入读写锁)是一个重要的同步工具,它属于Java并发包(java.util.concurrent.locks)中的一个类。这个锁提供了比标准的synchronized关键字更细粒度的控制,允许...
ReadWriteLock 接口及其实现 ReentrantReadWriteLock 方法 在 Java 并发编程中,锁机制是非常重要的一种同步机制,用于解决多线程之间的资源竞争问题。在 Java 中,有多种锁机制,如 ReentrantLock、...
在学习Java过程中,自己收集了很多的Java的学习资料,分享给大家,有需要的欢迎下载,希望对大家有用,一起学习,一起进步。
Java的ReentrantReadWriteLock是Java并发包`java.util.concurrent.locks`中的一个重要工具,它提供了一种更细粒度的锁定机制,相比普通的独占锁(如ReentrantLock)在某些场景下能显著提高程序的并发性能。...
6.5 深入理解 AQS之 ReentrantReadWritelock 实战副本.mp4
6.5 深入理解 AQS之 ReentrantReadWritelock 实战副本副本.mp4
ReentrantReadWriteLock 读写锁除了保证写操作对读操作可见性以及并发行提升外,简化了读写交互场景开发
ReadWriteLock的使用,实际上由于ReadWriteLock是一个接口,所以实际使用的是ReentrantReadWriteLock子类。同时ReadWriteLock的使用其实也是比较简单的,就是读写的锁的使用以及注意事项而已。
`ReentrantLock`和`ReentrantReadWriteLock`是Java并发包`java.util.concurrent.locks`中的两个重要工具,它们提供了比标准`synchronized`关键字更高级别的锁控制机制。本文将深入探讨这两个类,以及如何通过配置...
本篇文章主要介绍了Java concurrency之共享锁和ReentrantReadWriteLock,非常具有实用价值,需要的朋友可以参考下
ReentrantReadWriteLock reentrantReadWriteLock = new ReentrantReadWriteLock(); ``` 然后通过以下方式获取读锁和写锁: ```java ReentrantReadWriteLock.ReadLock readLock = reentrantReadWriteLock.readLock...
本文将深入探讨Java中的两种读写锁:ReentrantReadWriteLock和StampedLock,并分析它们的工作原理、特点以及如何在实际开发中进行应用。 一、ReentrantReadWriteLock(可重入读写锁) 1. **简介**: ...
内容包括 01-并发编程之深入理解JMM&并发三大特性(一)-fox 02-并发编程之深入理解JMM&并发三...11-深入理解AQS之CyclicBarrier&ReentrantReadWriteLock详解-fox 12-深入理解AQS之ReentrantReadWriteLock详解-fox ...
ReentrantReadWriteLock lock = new ReentrantReadWriteLock(); ReadLock readLock = lock.readLock(); WriteLock writeLock = lock.writeLock(); ``` 2. **读者部分**:当一个线程想要读取数据时,它需要获取读锁...
ReentrantReadWriteLock lock = new ReentrantReadWriteLock(true); // 公平锁 ``` 然后,我们可以定义读操作和写操作的方法: 对于读者: ```java public void read() { lock.readLock().lock(); try { // 读取...
- `ReentrantReadWriteLock`是Java提供的可重入的读写锁实现,它继承自`AbstractQueuedSynchronizer`(AQS)。可重入意味着一个线程可以多次获取同一锁,这在递归调用或者锁嵌套时很有用。 - **公平性**:`...