读写锁设计模式


前言

承接上一节讲单线程执行设计模式中例子的问题:读和写用的是同一个锁。当多个线程同时需要进行读操作的时候,只有一个线程能够读,其它线程被阻塞。导致读操作的串行化,这会使得程序的执行效率降低。为了解决这个问题,我们要将读操作并行化。于是就有了ReadWriteLock Design Pattern(读写锁设计模式)。

一、什么是读写锁设计模式

读写锁设计模式就是将读锁和写操作分离开来,主要是为了使得读操作并行化,多个线程可以同时进行读操作,以提高代码的执行效率。

二、问题考虑

在正式讲读写锁设计模式之前,我们先来考虑一个问题:读操作什么时候都进行吗?
答案是否定的,当有别的线程在进行写操作,这时我们不能进行读操作。只有当前没有任何线程在进行写操作的时候我们才能进行读操作。其它情况需要按照如下图去严格执行。不然写出的程序就不是线程安全的,这时候,程序执行效率再高都没有意义了。
在这里插入图片描述

三、读写锁设计

上图Y/N带表的是:是否不允许。可以这样理解,当有线程在读的时候,别的线程是可以进行读操作,但是别的线程不能进行写操作。当有线程在写的时候,别的线程既不能进行读操作也不能进行写操作。
首先我们要定义以下几个变量。
waitingReaders:正在等待读的线程的数量
readingReaders:正在进行读操作的线程的数量
waitingWriters:正在等待写的线程的数量
writingWriters:正在写的线程的数量

当有线程需要进行读操作的时候,这时就要拿到读锁。因为可能还没有获取到读锁,所以waitingReaders++,如果这时有别的线程在进行写操作,当前线程就不能进行读操作,要不断调用wait()方法进入阻塞状态,直到当前进行读操作的线程数writingWriters为0为止。最终线程能进行读操作,waitingReaders–

当有线程要进行写操作的时候,这时要去抢写锁,同样因为可能还没有获取到写锁,所以waitingWriters++,如果这时有别的线程在进行读操作或者有别的线程在进行写操作,当前线程就不能进行写操作,要不断调用wait()方法进入阻塞状态,直到当前的readingReaders和writingWriters都为0为止。最终现场能够进行写操作,所以waitingWriters–

当有线程要释放读锁的时候,readingReaders–,然后要调用notifyAll()方法。
当有线程要释放写锁的时候,writingWriters–,然后要调用notifyAll()方法。

代码如下:

package readwritelock;

public class ReadWriteLock {
    private int waitingReaders = 0;
    private int readingReaders = 0;
    private int waitingWriters = 0;
    private int writingWriters = 0;
    private boolean preferWriter = true;

    public ReadWriteLock () {
        this(true);
    }

    public ReadWriteLock (boolean preferWriter) {
        this.preferWriter = preferWriter;
    }

    public synchronized void readLock() {
        waitingReaders++;
        try {
            while (writingWriters > 0 || (preferWriter && waitingWriters > 0)) {
                this.wait();
            }
            readingReaders++;
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            waitingReaders--;
        }
    }

    public synchronized void writeLock() {
        waitingWriters++;
        try {
            while (readingReaders > 0 || writingWriters > 0) {
                this.wait();
            }
            writingWriters++;
        } catch (InterruptedException e) {
            e.printStackTrace();
        } finally {
            waitingWriters--;
        }
    }

    public synchronized void readUnLock() {
        readingReaders--;
        this.notifyAll();
    }

    public synchronized void writeUnLock() {
        waitingWriters--;
        this.notifyAll();
    }
}

四、需求分析

我们需求是多个线程同时对buffer缓冲进行读和写操作,应用读写锁设计模式。提高读效率,保证线程安全。

1、共享资源

这里要定义一个资源共享类,共享资源就是buffer缓冲区,这里使用一个char[],共享资源要提供读和写的方法供线程调用。重要的点就是怎么读和怎么写了:先拿到读锁或者写锁,然后进行读或者写操作,最后释放锁

代码如下:

package readwritelock;

public class SharedData {
    private final char[] buffer;
    private static final ReadWriteLock LOCK = new ReadWriteLock();
    public SharedData(int size) {
        this.buffer = new char[size];
        for (int i = 0; i < size; i++) {
            this.buffer[i] = '*';
        }
    }

    public char[] read() {
        LOCK.readLock();
        char[] newBuffer = doRead();
        LOCK.readUnLock();
        return newBuffer;
    }

    private char[] doRead() {
        char[] newBuffer = new char[buffer.length];
        for (int i = 0; i < buffer.length ; i++)
            newBuffer[i] = buffer[i];
        slowly(50);
        return newBuffer;
    }

    public void writer(char c) {
        LOCK.writeLock();
        doWrite(c);
        LOCK.writeUnLock();
    }

    private void doWrite(char c) {
        for (int i = 0; i <buffer.length ; i++)
            buffer[i] = c;
        slowly(10);
    }

    private void slowly(int ms) {
        try {
            Thread.sleep(ms);
        } catch (InterruptedException e) {
        }
    }
}

2、写工作线程

写工作线程,首先要对共享资源进行操作,那么就需要有共享资源的属性,线程启动,就对buffer进行写。

package readwritelock;

import java.util.Random;

public class WriterWorker extends Thread {
    private final SharedData data;
    private final String filler;
    private final Random random = new Random(System.currentTimeMillis());
    private int index = 0;

    public WriterWorker(SharedData data, String filler) {
        this.data = data;
        this.filler = filler;
    }

    @Override
    public void run() {
        try {
            while (true) {
                char c = nextChar();
                data.writer(c);
                Thread.sleep(random.nextInt(1000));
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    private char nextChar() {
        char c = filler.charAt(index);
        index++;
        if (index > filler.length()) {
            index = 0;
        }
        return c;
    }
}

3、读工作线程

与写线程一样,需要有共享资源属性

package readwritelock;
public class ReaderWorker extends Thread {

    private final SharedData data;

    public ReaderWorker(SharedData data) {
        this.data = data;
    }

    @Override
    public void run() {
        try {
            while (true) {
                char[] readBuf = data.read();
                System.out.println(Thread.currentThread().getName() + " reads " + String.valueOf(readBuf));
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

4、客户端

package readwritelock;

/**
 * ReadWriteLock design pattern
 * Reader-Writer design pattern
 */
public class ReadWritLockClient {
    public static void main(String[] args) {
        final SharedData sharedData = new SharedData(10);
        new ReaderWorker(sharedData).start();
        new ReaderWorker(sharedData).start();
        new ReaderWorker(sharedData).start();
        new ReaderWorker(sharedData).start();
        new ReaderWorker(sharedData).start();
        new WriterWorker(sharedData, "qwertyuiopasdfg").start();
        new WriterWorker(sharedData, "QWERTYUIOPASDFG").start();
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值