8.ReadWriteLock(读写锁)

感谢秦疆老师的JUC并发编程视频,更多了解哔哩哔哩搜索【狂神说Java】。

本文内容源于秦疆老师的JUC并发编程视频教程。给狂神推荐,点赞吧!

ReadWriteLock(读写锁 )

在这里插入图片描述

在这里插入图片描述

  • 独占锁(写锁) 一次只能被一个线程占有

  • 共享锁(读锁) 多个线程可以同时占有

  • 读写锁 读的时候可以被多线程同时读,写的时候只能有一个线程去写

代码测试:实现如下

  • 读-读 可以共存!
  • 读-写 不能共存!
  • 写-写 不能共存!
package demo6;

import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

/**
 *
 * 独占锁(写锁) 一次只能被一个线程占有
 * 共享锁(读锁) 多个线程可以同时占有
 * 读写锁 读的时候可以被多线程同时读,写的时候只能有一个线程去写
 * ReadWriteLock
 *
 * 读-读 可以共存!
 * 读-写 不能共存!
 * 写-写 不能共存!
 *
 *
 */
public class ReadWriteLockTest {
    public static void main(String[] args) {

      //  MyCache myCache = new MyCache();

        MyCacheLock myCacheLock = new MyCacheLock();
        //写入
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{
                myCacheLock.put(temp+"",temp+"");
            },String.valueOf(i)).start();
        }


        //读取
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{

                myCacheLock.get(temp+"");
            },String.valueOf(i)).start();
        }


    }

}

/**
 * 自定义缓存
 */
class MyCache {

    private volatile Map<String, Object> map = new HashMap<>();

    //存,写
    public void put(String key, Object value) {

        System.out.println(Thread.currentThread().getName() + "写入" + key);
        map.put(key,value);
        System.out.println(Thread.currentThread().getName() + "写入成功!" );

    }

    //取,读
    public void get(String key) {
        System.out.println(Thread.currentThread().getName() + "读取" + key);
        Object object = map.get(key);
        System.out.println(Thread.currentThread().getName() + "读取成功!");

    }


}


/**
 * 加锁的自定义缓存
 */
class MyCacheLock {

    private volatile Map<String, Object> map = new HashMap<>();

    //读写锁:更细粒度的控制
    private ReadWriteLock readWriteLock = new ReentrantReadWriteLock();

    //之前的锁 粗粒度的控制
    private Lock lock = new ReentrantLock();

    //存,写入的时候,只希望同时只有一个线程写
    public void put(String key, Object value) {

        //写锁
        readWriteLock.writeLock().lock();

        try {
            System.out.println(Thread.currentThread().getName() + "写入" + key);
            map.put(key,value);
            System.out.println(Thread.currentThread().getName() + "写入成功!" );

        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            readWriteLock.writeLock().unlock();
        }

    }

    //取,读,所有人都可以读
    public void get(String key) {

        readWriteLock.readLock().lock();

        try {
            System.out.println(Thread.currentThread().getName() + "读取" + key);
            Object object = map.get(key);
            System.out.println(Thread.currentThread().getName() + "读取成功!");

        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            readWriteLock.readLock().unlock();
        }

    }


}

未加读写锁的时候输出:

		 MyCache myCache = new MyCache();

        //写入
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{
                myCache.put(temp+"",temp+"");
            },String.valueOf(i)).start();
        }

        //读取
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{
                myCache.get(temp+"");
            },String.valueOf(i)).start();
        }


//混乱 写入插队
2写入2
5写入5
4写入4
1写入1
3写入3
1写入成功!
4写入成功!
5写入成功!
2写入成功!
3写入成功!
1读取1
2读取2
3读取3
4读取4
1读取成功!
2读取成功!
4读取成功!
3读取成功!
5读取5
5读取成功!

加了读写锁的时候输出:

  MyCacheLock myCacheLock = new MyCacheLock();
        //写入
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{
                myCacheLock.put(temp+"",temp+"");
            },String.valueOf(i)).start();
        }


        //读取
        for (int i = 1; i <= 5; i++) {
            final int temp = i;
            new Thread(()->{
                myCacheLock.get(temp+"");
            },String.valueOf(i)).start();
        }

//写入成功之后 
3写入3
3写入成功!
2写入2
2写入成功!
4写入4
4写入成功!
5写入5
5写入成功!
1写入1
1写入成功!
1读取1
1读取成功!
4读取4
5读取5
3读取3
2读取2
3读取成功!
5读取成功!
4读取成功!
2读取成功!
posted @ 2020-05-24 23:50  我有满天星辰  阅读(2)  评论(0编辑  收藏  举报