欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java多線(xiàn)程編程之讀寫(xiě)鎖ReadWriteLock用法實(shí)例

 更新時(shí)間:2015年05月19日 10:16:39   投稿:junjie  
這篇文章主要介紹了Java多線(xiàn)程編程之讀寫(xiě)鎖ReadWriteLock用法實(shí)例,本文直接給出編碼實(shí)例,需要的朋友可以參考下

讀寫(xiě)鎖:分為讀鎖和寫(xiě)鎖,多個(gè)讀鎖不互斥,讀鎖與寫(xiě)鎖互斥,這是由jvm自己控制的,你只要上好相應(yīng)的鎖即可。如果你的代碼只讀數(shù)據(jù),可以很多人同時(shí)讀,但不能同時(shí)寫(xiě),那就上讀鎖;如果你的代碼修改數(shù)據(jù),只能有一個(gè)人在寫(xiě),且不能同時(shí)讀取,那就上寫(xiě)鎖??傊?,讀的時(shí)候上讀鎖,寫(xiě)的時(shí)候上寫(xiě)鎖!

三個(gè)線(xiàn)程讀數(shù)據(jù),三個(gè)線(xiàn)程寫(xiě)數(shù)據(jù)示例:
可以同時(shí)讀,讀的時(shí)候不能寫(xiě),不能同時(shí)寫(xiě),寫(xiě)的時(shí)候不能讀。
讀的時(shí)候上讀鎖,讀完解鎖;寫(xiě)的時(shí)候上寫(xiě)鎖,寫(xiě)完解鎖。
注意finally解鎖。

package com.ljq.test.thread;
 
import java.util.Random;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
 
/**
 * 讀寫(xiě)鎖
 *
 * @author Administrator
 *
 */
public class ReadWriteLockTest {
  public static void main(String[] args) {
    final ReadWrite rw = new ReadWrite();
    for (int i = 0; i < 3; i++) {
      new Thread() {
        public void run() {
          while (true) {
            rw.read();
          }
        }
 
      }.start();
 
      new Thread() {
        public void run() {
          while (true) {
            rw.write(new Random().nextInt(10000));
          }
        }
 
      }.start();
    }
 
  }
}
 
/**
 * 讀和寫(xiě)要互斥,因此要把它們放在同一個(gè)類(lèi)中
 *
 * @author Administrator
 *
 */
class ReadWrite {
  private Object data = null;//共享數(shù)據(jù),只能有一個(gè)線(xiàn)程寫(xiě)該數(shù)據(jù),但可以有多個(gè)線(xiàn)程同時(shí)讀該數(shù)據(jù)。
  ReadWriteLock rwl = new ReentrantReadWriteLock();
 
  /**
   * 讀數(shù)據(jù)
   */
  public void read() {
     
    rwl.readLock().lock();
    try {
      System.out.println(Thread.currentThread().getName() + " be ready to read data!");
      Thread.sleep((long) (Math.random() * 1000));
      System.out.println(Thread.currentThread().getName() + "have read data :" + data);
    } catch (InterruptedException e) {
      e.printStackTrace();
    } finally {
      rwl.readLock().unlock();
    }
     
  }
 
  /**
   * 寫(xiě)數(shù)據(jù)
   *
   * @param data
   */
  public void write(Object data) {
     
    rwl.writeLock().lock();
    try {
      System.out.println(Thread.currentThread().getName() + " be ready to write data!");
      Thread.sleep((long) (Math.random() * 1000));
      this.data = data;
      System.out.println(Thread.currentThread().getName() + " have write data: " + data);
    } catch (InterruptedException e) {
      e.printStackTrace();
    } finally {
      rwl.writeLock().unlock();
    }
 
  }
}



設(shè)計(jì)一個(gè)緩存系統(tǒng)
緩存系統(tǒng):你要取數(shù)據(jù),需調(diào)用我的public Object getData(String key)方法,我要檢查我內(nèi)部有沒(méi)有這個(gè)數(shù)據(jù),如果有就直接返回,如果沒(méi)有,就從數(shù)據(jù)庫(kù)中查找這個(gè)數(shù),查到后將這個(gè)數(shù)據(jù)存入我內(nèi)部的存儲(chǔ)器中,下次再有人來(lái)要這個(gè)數(shù)據(jù),我就直接返回這個(gè)數(shù)不用再到數(shù)據(jù)庫(kù)中找了。你要取數(shù)據(jù)不要找數(shù)據(jù)庫(kù),來(lái)找我。

package com.ljq.test.thread;
 
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;
 
/**
 * 設(shè)計(jì)一個(gè)緩存系統(tǒng)
 *
 *
 * @author Administrator
 *
 */
public class CacheDemo {
 
  private Map<String, Object> cache = new HashMap<String, Object>();
 
  public static void main(String[] args) {
    String key = "name";
    CacheDemo cacheDemo = new CacheDemo();
    System.out.println(cacheDemo.getData(key)); //從數(shù)據(jù)庫(kù)獲取數(shù)據(jù)
    System.out.println(cacheDemo.getData(key)); //從緩存獲取數(shù)據(jù)
    System.out.println(cacheDemo.getData(key)); //從緩存獲取數(shù)據(jù)
  }
 
  private ReadWriteLock rwl = new ReentrantReadWriteLock();
 
  public Object getData(String key) {
    rwl.readLock().lock(); //上讀鎖
    Object value = null;
    try {
      value = cache.get(key); //先查詢(xún)內(nèi)部存儲(chǔ)器中有沒(méi)有要的值
      if (value == null) { //如果沒(méi)有,就去數(shù)據(jù)庫(kù)中查詢(xún),并將查到的結(jié)果存入內(nèi)部存儲(chǔ)器中
        //釋放讀鎖、上寫(xiě)鎖
        rwl.readLock().unlock();
        rwl.writeLock().lock();
        try {
          if (value == null) { //再次進(jìn)行判斷,防止多個(gè)寫(xiě)線(xiàn)程堵在這個(gè)地方重復(fù)寫(xiě)
            System.out.println("read data from database");
            value = "張三";
            cache.put(key, value);
          }
        } finally {
          //設(shè)置完成 釋放寫(xiě)鎖
          rwl.writeLock().unlock();
        }
        //恢復(fù)讀寫(xiě)狀態(tài)
        rwl.readLock().lock();
      }else{
        System.out.println("read data from cache");
      }
    } finally {
      rwl.readLock().unlock(); //釋放讀鎖
    }
    return value;
  }
}

返回結(jié)果:

相關(guān)文章

  • Java中的CountDownLatch原理深入解析

    Java中的CountDownLatch原理深入解析

    這篇文章主要介紹了Java中的CountDownLatch原理深入解析,CountDownLatch是多線(xiàn)程控制的一種同步工具類(lèi),它被稱(chēng)為門(mén)閥、 計(jì)數(shù)器或者閉鎖,這個(gè)工具經(jīng)常用來(lái)用來(lái)協(xié)調(diào)多個(gè)線(xiàn)程之間的同步,或者說(shuō)起到線(xiàn)程之間的通信,需要的朋友可以參考下
    2024-01-01
  • Spring?IOC中的組件掃描

    Spring?IOC中的組件掃描

    通過(guò)自動(dòng)掃描,Spring?會(huì)自動(dòng)從掃描指定的包及其子包下的所有類(lèi),并根據(jù)類(lèi)上的特定注解將該類(lèi)裝配到容器中,而無(wú)需在?XML?配置文件或?Java?配置類(lèi)中逐一聲明每一個(gè)?Bean,這篇文章主要介紹了Spring?IOC中的組件掃描,需要的朋友可以參考下
    2022-05-05
  • Java 中 synchronized的用法詳解(四種用法)

    Java 中 synchronized的用法詳解(四種用法)

    Java語(yǔ)言的關(guān)鍵字,當(dāng)它用來(lái)修飾一個(gè)方法或者一個(gè)代碼塊的時(shí)候,能夠保證在同一時(shí)刻最多只有一個(gè)線(xiàn)程執(zhí)行該段代碼。本文給大家介紹java中 synchronized的用法,對(duì)本文感興趣的朋友一起看看吧
    2015-11-11
  • SpringBoot2.3定制錯(cuò)誤頁(yè)面的方法示例

    SpringBoot2.3定制錯(cuò)誤頁(yè)面的方法示例

    這篇文章主要介紹了SpringBoot2.3定制錯(cuò)誤頁(yè)面的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-08-08
  • SpringBoot中熔斷器的原理和使用詳解

    SpringBoot中熔斷器的原理和使用詳解

    這篇文章主要介紹了SpringBoot中熔斷器的原理和使用詳解,熔斷器是一種用于處理分布式系統(tǒng)中故障的設(shè)計(jì)模式,它可以防止出現(xiàn)故障的服務(wù)對(duì)整個(gè)系統(tǒng)造成連鎖反應(yīng),需要的朋友可以參考下
    2023-07-07
  • springboot結(jié)合maven實(shí)現(xiàn)多模塊打包

    springboot結(jié)合maven實(shí)現(xiàn)多模塊打包

    本文主要介紹了springboot借助maven完成多模塊打包,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • 淺談Spring AOP中args()和argNames的含義

    淺談Spring AOP中args()和argNames的含義

    這篇文章主要介紹了Spring AOP中args()和argNames的含義,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • SpringBoot集成ip2region實(shí)現(xiàn)ip白名單的代碼示例

    SpringBoot集成ip2region實(shí)現(xiàn)ip白名單的代碼示例

    ip2region v2.0 - 是一個(gè)離線(xiàn)IP地址定位庫(kù)和IP定位數(shù)據(jù)管理框架,10微秒級(jí)別的查詢(xún)效率,提供了眾多主流編程語(yǔ)言的 xdb 數(shù)據(jù)生成和查詢(xún)客戶(hù)端實(shí)現(xiàn),本文介紹了SpringBoot集成ip2region實(shí)現(xiàn)ip白名單的代碼工程,需要的朋友可以參考下
    2024-08-08
  • Spring boot JPA實(shí)現(xiàn)分頁(yè)和枚舉轉(zhuǎn)換代碼示例

    Spring boot JPA實(shí)現(xiàn)分頁(yè)和枚舉轉(zhuǎn)換代碼示例

    這篇文章主要介紹了Spring boot JPA實(shí)現(xiàn)分頁(yè)和枚舉轉(zhuǎn)換代碼示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-09-09
  • Spring 項(xiàng)目常用pom文件的依賴(lài)

    Spring 項(xiàng)目常用pom文件的依賴(lài)

    這篇文章主要介紹了Spring 項(xiàng)目常用pom文件的依賴(lài),文中給大家提到了Spring boot starter pom的依賴(lài)關(guān)系說(shuō)明,需要的朋友參考下吧
    2018-03-03

最新評(píng)論