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

Java構(gòu)建高效結(jié)果緩存方法示例

 更新時(shí)間:2020年04月24日 11:27:03   作者:flydean程序那些事  
這篇文章主要介紹了Java構(gòu)建高效結(jié)果緩存方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

緩存是現(xiàn)代應(yīng)用服務(wù)器中非常常用的組件。除了第三方緩存以外,我們通常也需要在java中構(gòu)建內(nèi)部使用的緩存。那么怎么才能構(gòu)建一個(gè)高效的緩存呢? 本文將會(huì)一步步的進(jìn)行揭秘。

使用HashMap

緩存通常的用法就是構(gòu)建一個(gè)內(nèi)存中使用的Map,在做一個(gè)長(zhǎng)時(shí)間的操作比如計(jì)算之前,先在Map中查詢一下計(jì)算的結(jié)果是否存在,如果不存在的話再執(zhí)行計(jì)算操作。

我們定義了一個(gè)代表計(jì)算的接口:

public interface Calculator<A, V> {
  V calculate(A arg) throws InterruptedException;
}

該接口定義了一個(gè)calculate方法,接收一個(gè)參數(shù),并且返回計(jì)算的結(jié)果。

我們要定義的緩存就是這個(gè)Calculator具體實(shí)現(xiàn)的一個(gè)封裝。

我們看下用HashMap怎么實(shí)現(xiàn):

public class MemoizedCalculator1<A, V> implements Calculator<A, V> {

  private final Map<A, V> cache= new HashMap<A, V>();
  private final Calculator<A, V> calculator;
  public MemoizedCalculator1(Calculator<A, V> calculator){
    this.calculator=calculator;
  }
  @Override
  public synchronized V calculate(A arg) throws InterruptedException {
    V result= cache.get(arg);
    if( result ==null ){
      result= calculator.calculate(arg);
      cache.put(arg, result);
    }
    return result;
  }
}

MemoizedCalculator1封裝了Calculator,在調(diào)用calculate方法中,實(shí)際上調(diào)用了封裝的Calculator的calculate方法。

因?yàn)镠ashMap不是線程安全的,所以這里我們使用了synchronized關(guān)鍵字,從而保證一次只有一個(gè)線程能夠訪問(wèn)calculate方法。

雖然這樣的設(shè)計(jì)能夠保證程序的正確執(zhí)行,但是每次只允許一個(gè)線程執(zhí)行calculate操作,其他調(diào)用calculate方法的線程將會(huì)被阻塞,在多線程的執(zhí)行環(huán)境中這會(huì)嚴(yán)重影響速度。從而導(dǎo)致使用緩存可能比不使用緩存需要的時(shí)間更長(zhǎng)。

使用ConcurrentHashMap

因?yàn)镠ashMap不是線程安全的,那么我們可以嘗試使用線程安全的ConcurrentHashMap來(lái)替代HashMap。如下所示:

public class MemoizedCalculator2<A, V> implements Calculator<A, V> {

  private final Map<A, V> cache= new ConcurrentHashMap<>();
  private final Calculator<A, V> calculator;
  public MemoizedCalculator2(Calculator<A, V> calculator){
    this.calculator=calculator;
  }
  @Override
  public V calculate(A arg) throws InterruptedException {
    V result= cache.get(arg);
    if( result ==null ){
      result= calculator.calculate(arg);
      cache.put(arg, result);
    }
    return result;
  }
}

上面的例子中雖然解決了之前的線程等待的問(wèn)題,但是當(dāng)有兩個(gè)線程同時(shí)在進(jìn)行同一個(gè)計(jì)算的時(shí)候,仍然不能保證緩存重用,這時(shí)候兩個(gè)線程都會(huì)分別調(diào)用計(jì)算方法,從而導(dǎo)致重復(fù)計(jì)算。

我們希望的是如果一個(gè)線程正在做計(jì)算,其他的線程只需要等待這個(gè)線程的執(zhí)行結(jié)果即可。很自然的,我們想到了之前講到的FutureTask。FutureTask表示一個(gè)計(jì)算過(guò)程,我們可以通過(guò)調(diào)用FutureTask的get方法來(lái)獲取執(zhí)行的結(jié)果,如果該執(zhí)行正在進(jìn)行中,則會(huì)等待。

下面我們使用FutureTask來(lái)進(jìn)行改寫。

FutureTask

@Slf4j
public class MemoizedCalculator3<A, V> implements Calculator<A, V> {

  private final Map<A, Future<V>> cache= new ConcurrentHashMap<>();
  private final Calculator<A, V> calculator;

  public MemoizedCalculator3(Calculator<A, V> calculator){
    this.calculator=calculator;
  }
  @Override
  public V calculate(A arg) throws InterruptedException {
    Future<V> future= cache.get(arg);
    V result=null;
    if( future ==null ){
      Callable<V> callable= new Callable<V>() {
        @Override
        public V call() throws Exception {
          return calculator.calculate(arg);
        }
      };
      FutureTask<V> futureTask= new FutureTask<>(callable);
      future= futureTask;
      cache.put(arg, futureTask);
      futureTask.run();
    }
    try {
      result= future.get();
    } catch (ExecutionException e) {
      log.error(e.getMessage(),e);
    }
    return result;
  }
}

上面的例子,我們用FutureTask來(lái)封裝計(jì)算,并且將FutureTask作為Map的value。

上面的例子已經(jīng)體現(xiàn)了很好的并發(fā)性能。但是因?yàn)閕f語(yǔ)句是非原子性的,所以對(duì)這一種先檢查后執(zhí)行的操作,仍然可能存在同一時(shí)間調(diào)用的情況。

這個(gè)時(shí)候,我們可以借助于ConcurrentHashMap的原子性操作putIfAbsent來(lái)重寫上面的類:

@Slf4j
public class MemoizedCalculator4<A, V> implements Calculator<A, V> {

  private final Map<A, Future<V>> cache= new ConcurrentHashMap<>();
  private final Calculator<A, V> calculator;

  public MemoizedCalculator4(Calculator<A, V> calculator){
    this.calculator=calculator;
  }
  @Override
  public V calculate(A arg) throws InterruptedException {
    while (true) {
      Future<V> future = cache.get(arg);
      V result = null;
      if (future == null) {
        Callable<V> callable = new Callable<V>() {
          @Override
          public V call() throws Exception {
            return calculator.calculate(arg);
          }
        };
        FutureTask<V> futureTask = new FutureTask<>(callable);
        future = cache.putIfAbsent(arg, futureTask);
        if (future == null) {
          future = futureTask;
          futureTask.run();
        }

        try {
          result = future.get();
        } catch (CancellationException e) {
          log.error(e.getMessage(), e);
          cache.remove(arg, future);
        } catch (ExecutionException e) {
          log.error(e.getMessage(), e);
        }
        return result;
      }
    }
  }
}

上面使用了一個(gè)while循環(huán),來(lái)判斷從cache中獲取的值是否存在,如果不存在則調(diào)用計(jì)算方法。

上面我們還要考慮一個(gè)緩存污染的問(wèn)題,因?yàn)槲覀冃薷牧司彺娴慕Y(jié)果,如果在計(jì)算的時(shí)候,計(jì)算被取消或者失敗,我們需要從緩存中將FutureTask移除。

本文的例子可以參考https://github.com/ddean2009/learn-java-concurrency/tree/master/MemoizedCalculate

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • yaml文件中${}語(yǔ)法取值方式

    yaml文件中${}語(yǔ)法取值方式

    在Spring Boot中,配置文件中的${test.aa}等占位符的值可以通過(guò)系統(tǒng)屬性或依賴的其他模塊來(lái)獲取,這意味著,可以通過(guò)JVM參數(shù)或者系統(tǒng)屬性來(lái)指定這些值,例如,通過(guò)在啟動(dòng)命令中添加-Dtest.aa=your_value或在代碼中通過(guò)
    2024-10-10
  • Java實(shí)現(xiàn)文件點(diǎn)擊沒(méi)反應(yīng)的方法

    Java實(shí)現(xiàn)文件點(diǎn)擊沒(méi)反應(yīng)的方法

    jsp頁(yè)面鏈接,點(diǎn)擊訪問(wèn)action用IO流去下載服務(wù)器上的文件,問(wèn)題是任憑怎么點(diǎn)擊都沒(méi)反應(yīng),日志也不報(bào)錯(cuò)。這篇文章給大家介紹Java實(shí)現(xiàn)文件點(diǎn)擊沒(méi)反應(yīng)的方法,需要的朋友參考下吧
    2018-07-07
  • 使用IDEA查看jar包及jar包的正確打開(kāi)方式

    使用IDEA查看jar包及jar包的正確打開(kāi)方式

    IDEA 是把 jar 包添加為 Libraries,然后展開(kāi)后即可查看,因?yàn)槭蔷幾g后的 class 文件,所以注釋是沒(méi)有的,今天小編給大家介紹下使用IDEA查看jar包及jar包的正確打開(kāi)方式,感興趣的朋友一起看看吧
    2023-07-07
  • java hasNext()使用實(shí)例解析

    java hasNext()使用實(shí)例解析

    這篇文章主要介紹了java hasNext()使用實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-09-09
  • 解決SpringMVC接收不到ajaxPOST參數(shù)的問(wèn)題

    解決SpringMVC接收不到ajaxPOST參數(shù)的問(wèn)題

    今天小編就為大家分享一篇解決SpringMVC接收不到ajaxPOST參數(shù)的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-08-08
  • 淺聊一下Spring?Security的使用方法

    淺聊一下Spring?Security的使用方法

    Spring?Security?是一個(gè)基于?Spring?框架的安全框架,提供了一套安全性認(rèn)證和授權(quán)的解決方案,用于保護(hù)?Web?應(yīng)用程序和服務(wù),接下來(lái)小編就和大家聊聊Spring?Security,感興趣的小伙伴跟著小編一起來(lái)看看吧
    2023-08-08
  • springboot集成schedule實(shí)現(xiàn)定時(shí)任務(wù)

    springboot集成schedule實(shí)現(xiàn)定時(shí)任務(wù)

    在項(xiàng)目開(kāi)發(fā)過(guò)程中,我們經(jīng)常需要執(zhí)行具有周期性的任務(wù)。通過(guò)定時(shí)任務(wù)可以很好的幫助我們實(shí)現(xiàn)。本篇文章主要介紹了springboot集成schedule實(shí)現(xiàn)定時(shí)任務(wù),感興趣的小伙伴們可以參考一下
    2018-05-05
  • 一文帶你掌握J(rèn)ava中的HashSet

    一文帶你掌握J(rèn)ava中的HashSet

    HashSet?基于?HashMap?來(lái)實(shí)現(xiàn)的,是一個(gè)不允許有重復(fù)元素的集合,HashSet?不是線程安全的,?如果多個(gè)線程嘗試同時(shí)修改?HashSet,則最終結(jié)果是不確定的,本文將帶你詳細(xì)了解Java中的HashSet,,需要的朋友可以參考下
    2023-05-05
  • java如何對(duì)map進(jìn)行排序詳解(map集合的使用)

    java如何對(duì)map進(jìn)行排序詳解(map集合的使用)

    這篇文章主要介紹了java如何對(duì)map進(jìn)行排序,java map集合的使用詳解,大家可以參考使用
    2013-12-12
  • 詳解SpringCloud eureka服務(wù)狀態(tài)監(jiān)聽(tīng)

    詳解SpringCloud eureka服務(wù)狀態(tài)監(jiān)聽(tīng)

    這篇文章主要介紹了詳解SpringCloud eureka服務(wù)狀態(tài)監(jiān)聽(tīng),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-07-07

最新評(píng)論