`

读写锁的使用

阅读更多

转自:http://www.blogjava.net/junglesong/archive/2008/02/22/181400.html

读写锁的使用

本文内容

何时该使用读写锁.
读写锁的写法.
理解读写锁和线程互斥的区别。

复习-同步化的概念

当一个方法或代码块被声明成synchronized,要执行此代码必须先取得一个对象实例或this的锁定,这个锁定要在synchronized修饰的方法或代码块执行完后才能释放掉(无论这段代码是怎样返回的,是正常运行还是异常运行)。每个对象只有一个锁定,如果有两个不同的线程试图同时调用同一对象的同步方法,最终只会有一个能运行此方法,另外一个要等待第一个线程释放掉锁定后才能运行此方法。

读写锁应用的场合

我们有时会遇到对同一个内存区域如数组或者链表进行多线程读写的情况,一般来说有以下几种处理方式: 1.不加任何限制,多见于读取写入都很快的情况,但有时也会出现问题. 2.对读写函数都加以同步互斥,这下问题是没了,但效率也下去了,比如说两个读取线程不是非要排队进入不可. 3.使用读写锁,安全和效率都得到了解决,特别合适读线程多于写线程的情况.也就是下面将要展现的模式.

读写锁的意图

读写锁的本意是分别对读写状态进行互斥区分,有互斥时才加锁,否则放行.互斥的情况有: 1.读写互斥. 2.写写互斥. 不互斥的情况是:读读,这种情况不该加以限制. 程序就是要让锁对象知道当前读写状态,再根据情况对读写的线程进行锁定和解锁。

读写线程都要操作的数据类

读写线程都要操作的数据是链表datas。
注意其中try...finally 的写法,它保证了加锁解锁过程是成对调用的

 

lpublic class DataLib {
     private List<String> datas;
 
     private ReadWriteLock lock;
 
     public DataLib() {
         datas = new ArrayList<String>();
         lock = new ReadWriteLock();
     }
 
     // 写入数据,这时不能读取
     public void writeData(List<String> newDatas) {
         try {
             lock.writeLock();
             Test.sleep(2);
             datas=newDatas;
         } finally {
             lock.writeUnlock();
         }
     }
 
     // 读取数据,这时不能写入
     public List<String> readData() {
         try {
             lock.readLock();
             Test.sleep(1);            
             return datas;
         } finally {
             lock.readUnlock();
         }
 
     }
 
 }


读写锁ReadWriteLock类

public class ReadWriteLock{
    // 读状态
    private boolean isRead;
    
    // 写状态
    private boolean isWrite;
    
    public synchronized void readLock(){
        // 有写入时读取线程停止
        while(isWrite){
            try{    
                System.out.println("有线程在进行写入,读取线程停止,进入等待状态");
                wait();
            }
            catch(InterruptedException ex){
                ex.printStackTrace();
            }
        }
        
        System.out.println("设定锁为读取状态");
        isRead=true;
    }
    
    public synchronized void readUnlock(){
        System.out.println("解除读取锁");
        isRead=false;
        notifyAll();
    }

     public synchronized void writeLock(){
        // 有读取时读取线程停止
        while(isRead){
            try{    
                System.out.println("有线程在进行读取,写入线程停止,进入等待状态");
                wait();
            }
            catch(InterruptedException ex){
                ex.printStackTrace();
            }
        }
        
        // 有写入时写入线程也一样要停止
        while(isWrite){
            try{    
                System.out.println("有线程在进行写入,写入线程停止,进入等待状态");
                wait();
            }
            catch(InterruptedException ex){
                ex.printStackTrace();
            }
        }
        
        System.out.println("设定锁为写入状态");
        isWrite=true;
    }
    
    public synchronized void writeUnlock(){
        System.out.println("解除写入锁");
        isWrite=false;
        notifyAll();
    }
}

写线程类Writer -它用于往DataLib类实例中的datas字段写数据

分析其中dataLib字段的用意。
注意并记住其中持续调用及使用随机数的方法。

 

lpublic class Writer implements Runnable{
     private DataLib dataLib;
     private static final Random random=new Random();
     private String[] mockDatas={"","","","","","","","","",""};    
     
     public Writer(DataLib dataLib,String[] mockDatas){
         this.dataLib=dataLib;
         this.mockDatas=mockDatas;
         
         Thread thread=new Thread(this);
         thread.start();
     }
     
     public void run(){
         while(true){
             Test.sleep(random.nextInt(3));
             
             int startIndex=random.nextInt(mockDatas.length);
             
             ArrayList<String> newDatas=new ArrayList<String>();
             for(int i=startIndex;i<mockDatas.length;i++){
                 newDatas.add(mockDatas[i]);
             }
             
             dataLib.writeData(newDatas);
         }
     }
 }

读线程类Reader  -它用于从DataLib类实例中的datas字段读取数据

分析其中dataLib字段的用意。
注意并记住其中持续调用及使用随机数的方法。

public class Reader implements Runnable{
    private DataLib dataLib;
    private static final Random random=new Random();
    
    public Reader(DataLib dataLib){
        this.dataLib=dataLib;
    
        Thread thread=new Thread(this);
        thread.start();
    }
    
    public void run(){
        while(true){
            Test.sleep(random.nextInt(2));            
            List<String> datas=dataLib.readData();
            
            System.out.print(">>取得数组为:");
            for(String data:datas){
                System.out.print(data+",");
            }
            System.out.print("\n");
        }
    }
}

将代码运行起来

右边的代码创建了两个写线程和三个读线程,它们都是对dataLib实例进行操作的。
五个线程都有一个dataLib字段,都提供了一个带参构造函数以给datas字段赋值,这就保证了五个线程操作的都是一个实例的同一字段,也就是同一片内存。
读写锁就是对这五个线程进行控制的。
当有一个读线程在操作时,其它的写线程无法进行操作,读线程可以正常操作,互不干扰。
当有一个写线程在操作时,其它的读线程无法进行操作。

 

 public class Test{
     public static void main(String[] args){
         DataLib dataLib=new DataLib();
         
         String[] mockDatas1={"","","","","","","","","",""};
         Writer writer1=new Writer(dataLib,mockDatas1);
         
         String[] mockDatas2={"","","","","","","","","","","",""};
         Writer writer2=new Writer(dataLib,mockDatas2);
         
         Reader reader1=new Reader(dataLib);
         Reader reader2=new Reader(dataLib);
         Reader reader3=new Reader(dataLib);
     }
     
     
     // 用于延时
     public static void sleep(int sleepSecond){
         try{
             Thread.sleep(sleepSecond*1000);
         }
         catch(Exception ex){
             ex.printStackTrace();
         }
     }
 }

小结

当多个线程试图对同一内容进行读写操作时适合使用读写锁。
请理解并记住ReadWriteLock类读写锁的写法.
读写锁相对于线程互斥的优势在于高效,它不会对两个读线程进行盲目的互斥处理,当读线程数量多于写线程尤其如此,当全是写线程时两者等效。

分享到:
评论

相关推荐

    读写锁_读写锁_

    下面是一个简单的读写锁使用示例: ```java import java.util.concurrent.locks.ReentrantReadWriteLock; import java.util.LinkedList; public class ReadWriteLockList { private final ReentrantReadWriteLock...

    WINDOWS读写锁实现

    在Windows操作系统中,读写锁(Read-Write Lock)是一种多线程同步原语,它允许多个线程同时读取共享资源,但在写入时仅允许一个线程访问。这提高了并发性能,尤其是在读操作远多于写操作的场景下。本篇文章将深入...

    读写锁实现代码

    一般的读写锁代码,都是使用了临界体作为基本锁,这样就出现了在一个线程里获取临界体,但是却在另一个线程里释放临界体的问题。临界体是线程相关锁,必须做到获取...这个读写锁使用临界体和事件的组合解决了这个问题。

    windows和linux读写锁C++实现

    本话题将详细探讨在Windows和Linux环境下,如何使用C++来实现读写锁。 首先,我们来看读写锁的基本概念。读写锁通常由两个部分组成:读锁和写锁。读锁允许多个线程同时读取数据,而写锁则确保在写操作进行时没有...

    windows 读写锁 (基于关键区的读写锁类 及自动锁)

    其中,读写锁(Read-Write Lock,简称RWLock)是一种高效的线程同步机制,适用于大量读取操作和少量写入操作的情况。在本文中,我们将深入探讨基于关键区的Windows读写锁类及其自动锁的实现原理和改进点。 读写锁的...

    java 读写锁代码

    - 在需要大量并发读取,偶尔修改的数据结构中,使用读写锁能有效提高程序性能。 - 在`readwritelock`这个示例中,可能包含了创建`ReentrantReadWriteLock`对象,获取读锁和写锁的代码,并通过注释解释了其工作原理...

    易语言线程安全之原子锁与读写锁

    在易语言中,我们可以使用读写锁类来实现这一功能。读写锁分为读锁和写锁,读锁可由多个线程同时持有,而写锁是独占的,一旦有线程持有了写锁,其他所有线程都无法再获取读锁或写锁。 读写锁的工作原理如下: 1. 当...

    linux写优先的读写锁设计

    在 Linux 内核中,读写锁的实现方式是通过使用魔数(magic)来标识锁的状态。当读操作请求锁时,如果锁已经被占用,读操作将等待直到锁释放。如果写操作请求锁时,如果锁已经被占用,写操作将等待直到锁释放。 读写...

    C#使用读写锁三行代码简单解决多线程并发的问题

    本文通过三行代码展示了如何使用读写锁解决多线程并发写入文件的冲突。 读写锁的核心在于,它允许多个线程同时读取资源,但只允许一个线程写入资源。这样,当多个线程尝试写入时,只有一个线程能够获得写锁,其他...

    易语言读写锁1.2模块源码

    易语言读写锁1.2模块源码例程程序调用API函数实现读写锁功能的操作。点评:读写锁实际是一种特殊的锁,它把对共享资源的访问者划分成读者和写者,读者只对共享资源进行读访问,写者则对共享资源进行写操作。这种锁相...

    读写锁实现例子

    在这个例子中,我们讨论了如何使用读写锁(Read-Write Lock)来实现多线程环境中的并发访问控制。读写锁是一种高级的同步机制,它允许多个线程同时读取共享资源,但只允许一个线程进行写操作。这在数据读取远比写入...

    操作系统实验--读写锁

    通过理解并正确使用读写锁,我们可以有效地提升多线程环境下的程序性能,同时保证数据的一致性。在进行操作系统实验时,理解并实践读写锁的工作原理,对于深入学习操作系统以及提升并发编程能力至关重要。

    C#解决SQlite并发异常问题的方法(使用读写锁)

    本文实例讲述了C#解决SQlite并发异常问题的方法。分享给大家供大家参考,...作者利用读写锁(ReaderWriterLock),达到了多线程安全访问的目标。 using System; using System.Collections.Generic; using System.Text;

    基于zookeeper实现的分布式读写锁

    本文将深入探讨基于Zookeeper实现的分布式读写锁,并利用Zkclient客户端进行操作。Zookeeper是一个分布式服务协调框架,它提供了一种简单且高效的方式来实现分布式锁。 **一、Zookeeper简介** Zookeeper是由Apache...

    易语言读写锁模块

    易语言读写锁模块源码,读写锁模块,进入,退出,读锁,写锁,解锁,InitializeCriticalSection,DeleteCriticalSection,EnterCriticalSection,LeaveCriticalSection

    读写锁源代码(C#编写)

    本文将深入探讨C#中的读写锁及其使用,帮助你理解和掌握这一关键技能。 首先,读写锁提供了对共享资源的细粒度控制。在C#中,`System.Threading.ReaderWriterLockSlim`类是实现读写锁的主要工具。它的设计目标是在...

    简单读写锁实例代码

    在C++编程中,读写锁(Read-Write Lock)是一种多线程同步机制,它允许多个线程同时读取共享资源,但只允许一个线程...在设计和使用读写锁时,应充分理解其工作原理,并结合具体场景进行优化,以实现最佳性能和安全性。

    linux 读写锁应用实例

    /*使用读写锁实现四个线程读写一段程序的实例,共创建了四个新的线程,其中两个线程用来读取数据,另外两个线程用来写入数据。在任意时刻,如果有一个线程在写数据,将阻塞所有其他线程的任何操作。*/

    Windows下读写锁的实现

    使用C++标准库中的`std::mutex`和`std::condition_variable`可以简化Windows下的读写锁实现。不过,题目提到的"RWLock"可能包含了使用原生Windows API(如CreateMutex和CreateSemaphore)的实现,这种实现更底层,...

Global site tag (gtag.js) - Google Analytics