读写锁

读写锁

读写锁要求:写入的时候只能有一个线程操作共享资源(一个线程写操作的时候不允许被插队,写完另一个线程再进行写操作),读的时候可以同时进行。这样做可以提升效率。

示例 不使用读写锁的时候

public class ReadWriteExample01 {
    public static void main(String[] args) {
        // 线程操作资源类
        MyCache myCache = new MyCache();
        for (int i = 1; i < 3; i++) {
            int finalI = i;
            new Thread(()->{
                myCache.write(String.valueOf(finalI),String.valueOf(finalI));
            },String.valueOf(i)).start();
        }
        for (int i = 1; i < 3; i++) {
            int finalI = i;
            new Thread(()->{
                myCache.read(String.valueOf(finalI));
            },String.valueOf(i)).start();
        }
    }
}

/*
 * 自定义缓存
 *
 * */
class MyCache{
    private volatile Map<String,String> map = new HashMap<>();

    //    存,写
    public void write(String key,String value){
        System.out.println(Thread.currentThread().getName()+"线程开始写入");
        map.put(key,value);
        System.out.println(Thread.currentThread().getName()+"线程开始写入ok");
    }
    //      取,读
    public void read(String key){
        System.out.println(Thread.currentThread().getName()+"线程开始读取");
        map.get(key);
        System.out.println(Thread.currentThread().getName()+"线程读取ok");
    }
}

结果 我们发现1线程还没有写入ok就被2线程插队了

1线程开始写入
2线程开始写入
2线程开始写入ok
1线程开始写入ok
2线程开始读取
1线程开始读取
1线程读取ok
2线程读取ok

示例 加入读写锁后

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

public class ReadWriteExample02 {
    public static void main(String[] args) {
        //线程操作资源类
        MyCache myCache = new MyCache();
        for (int i = 1; i < 3; i++) {
            int finalI = i;
            new Thread(()->{
                myCache.write(String.valueOf(finalI),String.valueOf(finalI));

            },String.valueOf(i)).start();
        }
        for (int i = 1; i < 3; i++) {
            int finalI = i;
            new Thread(()->{
                myCache.read(String.valueOf(finalI));
            },String.valueOf(i)).start();
        }
    }
}

/*
 * 自定义缓存
 *
 * */
class MyCache{
    private volatile Map<String,String> map = new HashMap<>();
    private ReadWriteLock lock = new ReentrantReadWriteLock();

    //    存,写
    public void write(String key,String value){
        lock.writeLock().lock();//写锁
        try {
            System.out.println(Thread.currentThread().getName()+"线程开始写入");
            map.put(key,value);
            System.out.println(Thread.currentThread().getName()+"线程开始写入ok");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.writeLock().unlock();
        }
    }
    //      取,读
    public void read(String key){
        lock.readLock().lock();
        try {
            System.out.println(Thread.currentThread().getName()+"线程开始读取");
            map.get(key);
            System.out.println(Thread.currentThread().getName()+"线程读取ok");
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            lock.readLock().unlock();
        }
    }
}

结果:我们发现在写入的时候保证了一个线程执行完成,才执行了另一个线程。

1线程开始写入
1线程开始写入ok
2线程开始写入
2线程开始写入ok
2线程开始读取
2线程读取ok
1线程开始读取
1线程读取ok

总结:

  • 独占锁(也叫写锁)一次只能被一个线程占有
  • 共享锁(也叫读锁)多个线程可以同时占有

参考教程:https://www.kuangstudy.com/

posted @ 2021-04-25 10:29  懒鑫人  阅读(88)  评论(0)    收藏  举报