JUC并发编程(九)- 读写锁ReadWriteLock详解

9、读写锁:ReadWriteLock

读写分离:读写分离是一种思想。

​ 之前讲的Lock是一个同步锁,它对资源的读和写,要么都锁住,要么都不锁,有的时候我们的资源,希望读的时候不锁,写的时候锁住,这样在安全的基础上能提高很大的效率。

在这里插入图片描述

​ 我们希望锁只添加在需要的地方,如在写的时候,加锁,只允许一个线程去操作,读的时候不加锁,所有的线程都可以去读。

在这里插入图片描述

这样ReadWriteLock运营而生。

在这里插入图片描述

读锁(共享锁): 这个锁可以被多个线程持有!

写锁(独占锁):这个锁一次只能被一个线程占用!

不使用锁的情况下,示例代码如下:

package com.interview.concurrent.readwritelock;

import java.util.HashMap;
import java.util.Map;

/**
 * @author yangxj
 * @description 描述
 * @date 2020/2/23 15:54
 */
public class ReadWriteLockDemo {
    public static void main(String[] args) {
        //不加锁
        ResourcePool resourcePool = new ResourcePool();

        //开启5个线程去写
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(() -> resourcePool.put(temp,temp),"写线程" + i).start();
        }
        //开启5个线程去读
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(() -> resourcePool.get(temp),"读线程" + i).start();
        }
    }
}

/**
 *  @description: 资源池,,没有加锁,资源池使用volatile,保证可见性
 *  @author yangxj
 *  @date 2020/2/23 16:07
 */
class ResourcePool{

    private volatile Map<Object,Object> map = new HashMap<Object,Object>();
    public void put(Object key,Object value){
        System.out.println(Thread.currentThread().getName() + ":写入数据" + key);
        map.put(key,value);
        System.out.println(Thread.currentThread().getName() + ":写入数据完成" + value);
    }

    public void get(Object key){
        System.out.println(Thread.currentThread().getName() + ":获取数据" + key);
        Object object = map.get(key);
        System.out.println(Thread.currentThread().getName() + ":获取数据完成" + object);
    }

}

在这里插入图片描述

使用读写锁后,线程之间就不会插队,一个线程必须等待其他线程完成才能执行。

ReadWriteLock编码模型:

1、创建读写锁

ReadWriteLock readWriteLock = new ReentrantReadWriteLock();

可使用两种锁,写锁和读锁

2.1、加写锁

readWriteLock.writeLock().lock();

2.2、业务代码:…

2.3、解写锁

try/catch/finally {
     readWriteLock.writeLock().unlock();
 }
 readWriteLock.readLock().lock();

3.1、加锁

readWriteLock.readLock().lock();

3.1、加锁

3.2、业务代码:…

3.3、解读锁

 try/catch/finally {
            readWriteLock.readLock().unlock();
   }

示例代码如下:

/**
 *  @description: 资源池,,没有加锁,资源池使用volatile,保证可见性
 *  @author yangxj
 *  @date 2020/2/23 16:07
 */
class ResourcePoolReadWriteLock{
    private volatile Map<Object,Object> map = new HashMap<Object,Object>();
    //1、创建读写锁
    ReadWriteLock readWriteLock = new ReentrantReadWriteLock();
    public void put(Object key,Object value){
        //2、加锁
        readWriteLock.writeLock().lock();

        try {
            System.out.println(Thread.currentThread().getName() + ":写入数据" + key);
            map.put(key,value);
            System.out.println(Thread.currentThread().getName() + ":写入数据完成" + value);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //3、解锁
            readWriteLock.writeLock().unlock();
        }
    }

    public void get(Object key){
        //2、加锁
        readWriteLock.readLock().lock();
        try {
            System.out.println(Thread.currentThread().getName() + ":获取数据" + key);
            Object object = map.get(key);
            System.out.println(Thread.currentThread().getName() + ":获取数据完成" + object);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            //3、解锁
            readWriteLock.readLock().unlock();
        }

    }
}

注:

1、线程是CPU调度的,与执行顺序无关;
2、ReadWriteLock的唯一实现类是ReentrantReadWriteLock
3、Lock不能区分读和写;
4、读写分离,提高效率~ 判断业务中那些代码是只读的业务,不要去锁这些业务。

发布了155 篇原创文章 · 获赞 23 · 访问量 11万+

猜你喜欢

转载自blog.csdn.net/makyan/article/details/104525015