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

Java本地緩存工具之LoadingCache的使用詳解

 更新時間:2021年12月31日 16:46:32   作者:劍客阿良_ALiang  
緩存,在我們?nèi)粘i_發(fā)中是必不可少的一種解決性能問題的方法。簡單的說,cache?就是為了提升系統(tǒng)性能而開辟的一塊內(nèi)存空間。本文將為大家介紹一個Java本地緩存的工具——LoadingCache,感興趣的可以了解一下

前言

在工作總常常需要用到緩存,而redis往往是首選,但是短期的數(shù)據(jù)緩存一般我們還是會用到本地緩存。本文提供一個我在工作中用到的緩存工具,該工具代碼為了演示做了一些調(diào)整。如果拿去使用的話,可以考慮做成注入Bean對象,看具體需求了。

環(huán)境依賴

先添加maven依賴

        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>30.1.1-jre</version>
        </dependency>
        <dependency>
            <groupId>cn.hutool</groupId>
            <artifactId>hutool-all</artifactId>
            <version>5.5.2</version>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>

代碼

不廢話,上代碼了。

package ai.guiji.csdn.tools;
 
import cn.hutool.core.thread.ThreadUtil;
import com.google.common.cache.*;
import lombok.extern.slf4j.Slf4j;
 
import java.text.MessageFormat;
import java.util.Map;
import java.util.concurrent.TimeUnit;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.LongStream;
 
/** @Author 劍客阿良_ALiang @Date 2021/12/30 17:57 @Description: 緩存工具 */
@Slf4j
public class CacheUtils {
 
  private static LoadingCache<Long, String> cache;
 
  /**
   * 初始化緩存方法
   *
   * @param totleCount 緩存池上限
   * @param overtime 超時時間
   * @param unit 時間單位
   * @param handleNotExist 處理不存在key方法
   * @param handleRemove 移除主鍵消費(fèi)
   */
  private static void initCache(
      Integer totleCount,
      Integer overtime,
      TimeUnit unit,
      Function<Long, String> handleNotExist,
      Consumer<Long> handleRemove) {
    cache =
        CacheBuilder.newBuilder()
            // 緩存池大小
            .maximumSize(totleCount)
            // 設(shè)置時間對象沒有被讀/寫訪問則對象從內(nèi)存中刪除
            .expireAfterWrite(overtime, unit)
            // 移除監(jiān)聽器
            .removalListener(
                new RemovalListener<Long, String>() {
                  @Override
                  public void onRemoval(RemovalNotification<Long, String> rn) {
                    handleRemove.accept(rn.getKey());
                  }
                })
            .recordStats()
            .build(
                new CacheLoader<Long, String>() {
                  @Override
                  public String load(Long aLong) throws Exception {
                    return handleNotExist.apply(aLong);
                  }
                });
    log.info("初始化緩存");
  }
 
  /**
   * 存入緩存
   *
   * @param key 鍵
   * @param value 值
   */
  public static void put(Long key, String value) {
    try {
      log.info("緩存存入:[{}]-[{}]", key, value);
      cache.put(key, value);
    } catch (Exception exception) {
      log.error("存入緩存異常", exception);
    }
  }
 
  /**
   * 批量存入緩存
   *
   * @param map 映射
   */
  public static void putMap(Map<Long, String> map) {
    try {
      log.info("批量緩存存入:[{}]", map);
      cache.putAll(map);
    } catch (Exception exception) {
      log.error("批量存入緩存異常", exception);
    }
  }
 
  /**
   * 獲取緩存
   *
   * @param key 鍵
   */
  public static String get(Long key) {
    try {
      return cache.get(key);
    } catch (Exception exception) {
      log.error("獲取緩存異常", exception);
      return null;
    }
  }
 
  /**
   * 刪除緩存
   *
   * @param key 鍵
   */
  public static void removeKey(Long key) {
    try {
      cache.invalidate(key);
    } catch (Exception exception) {
      log.error("刪除緩存異常", exception);
    }
  }
 
  /**
   * 批量刪除緩存
   *
   * @param keys 鍵
   */
  public static void removeAll(Iterable<Long> keys) {
    try {
      cache.invalidateAll(keys);
    } catch (Exception exception) {
      log.error("批量刪除緩存異常", exception);
    }
  }
 
  /** 清理緩存 */
  public static void clear() {
    try {
      cache.invalidateAll();
    } catch (Exception exception) {
      log.error("清理緩存異常", exception);
    }
  }
 
  /**
   * 獲取緩存大小
   *
   * @return 長度
   */
  public static long size() {
    return cache.size();
  }
 
  public static void main(String[] args) {
    initCache(
        Integer.MAX_VALUE,
        10,
        TimeUnit.SECONDS,
        k -> {
          log.info("緩存:[{}],不存在", k);
          return "";
        },
        x -> log.info("緩存:[{}],已經(jīng)移除", x));
    System.out.println(size());
    LongStream.range(0, 10).forEach(a -> put(a, MessageFormat.format("tt-{0}", a)));
    System.out.println(cache.asMap());
    ThreadUtil.sleep(5000);
    LongStream.range(0, 10)
        .forEach(
            a -> {
              System.out.println(get(a));
              ThreadUtil.sleep(1000);
            });
    System.out.println(cache.asMap());
    ThreadUtil.sleep(10000);
    System.out.println(cache.asMap());
  }
}

代碼說明

1、在初始化loadingCache的時候,可以添加緩存的最大數(shù)量、消逝時間、消逝或者移除監(jiān)聽事件、不存在鍵處理等等。在上面的代碼中,我初始化緩存大小為Integer的最大值,寫入10秒后消逝,如不存在key返回空字符串等等。

2、該類也提供了put、putAll、get、remove、removeAll、clear、size方法,可以對緩存進(jìn)行存、取、刪、清理、大小等操作。

3、main演示方法中,先往緩存存入10個數(shù)據(jù),然后過5秒后每秒取一個數(shù)據(jù),并且打印一下緩存中的全部內(nèi)容。

4、補(bǔ)充一句LoadingCache是線程安全的哦。

演示一下

15:31:53.495 [main] INFO ai.guiji.csdn.tools.CacheUtils - 初始化緩存
0
15:31:53.502 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[0]-[tt-0]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[1]-[tt-1]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[2]-[tt-2]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[3]-[tt-3]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[4]-[tt-4]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[5]-[tt-5]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[6]-[tt-6]
15:31:53.508 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[7]-[tt-7]
15:31:53.509 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[8]-[tt-8]
15:31:53.509 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存存入:[9]-[tt-9]
{6=tt-6, 5=tt-5, 0=tt-0, 8=tt-8, 7=tt-7, 2=tt-2, 1=tt-1, 9=tt-9, 3=tt-3, 4=tt-4}
tt-0
tt-1
tt-2
tt-3
tt-4
15:32:03.572 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[5],已經(jīng)移除
15:32:03.573 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[6],已經(jīng)移除
15:32:03.573 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[5],不存在
15:32:04.581 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[6],不存在
15:32:05.589 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[0],已經(jīng)移除
15:32:05.589 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[7],已經(jīng)移除
15:32:05.589 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[8],已經(jīng)移除
15:32:05.589 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[7],不存在
15:32:06.589 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[8],不存在
15:32:07.591 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[1],已經(jīng)移除
15:32:07.591 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[2],已經(jīng)移除
15:32:07.591 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[9],已經(jīng)移除
15:32:07.591 [main] INFO ai.guiji.csdn.tools.CacheUtils - 緩存:[9],不存在
{6=, 5=, 8=, 7=, 9=}
{}
Process finished with exit code 0

可以看到,后面的5-9在內(nèi)存中已經(jīng)不存在對應(yīng)的值了。

總結(jié)

本文提供的工具代碼主要是為了演示,實際工作中可以按照自己的需求做調(diào)整。

到此這篇關(guān)于Java本地緩存工具之LoadingCache的使用詳解的文章就介紹到這了,更多相關(guān)Java LoadingCache本地緩存內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • SpringBoot開發(fā)項目,引入JPA找不到findOne方法的解決

    SpringBoot開發(fā)項目,引入JPA找不到findOne方法的解決

    這篇文章主要介紹了SpringBoot開發(fā)項目,引入JPA找不到findOne方法的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • JavaWeb開發(fā)之使用jQuery與Ajax實現(xiàn)動態(tài)聯(lián)級菜單效果

    JavaWeb開發(fā)之使用jQuery與Ajax實現(xiàn)動態(tài)聯(lián)級菜單效果

    這篇文章主要介紹了JavaWeb開發(fā)之使用jQuery與Ajax實現(xiàn)動態(tài)聯(lián)級菜單效果的相關(guān)資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-10-10
  • java實現(xiàn)相同屬性名稱及相似類型的pojo、dto、vo等互轉(zhuǎn)操作

    java實現(xiàn)相同屬性名稱及相似類型的pojo、dto、vo等互轉(zhuǎn)操作

    這篇文章主要介紹了java實現(xiàn)相同屬性名稱及相似類型的pojo、dto、vo等互轉(zhuǎn)操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • Java流程控制語句最全匯總(中篇)

    Java流程控制語句最全匯總(中篇)

    這篇文章主要介紹了Java流程控制語句最全匯總(中篇),本文章內(nèi)容詳細(xì),通過案例可以更好的理解數(shù)組的相關(guān)知識,本模塊分為了三部分,本次為中篇,需要的朋友可以參考下
    2023-01-01
  • SpringBoot中Mybatis + Druid 數(shù)據(jù)訪問的詳細(xì)過程

    SpringBoot中Mybatis + Druid 數(shù)據(jù)訪問的詳細(xì)過程

    Spring Boot 底層都是采用 SpringData 的方式進(jìn)行統(tǒng)一處理各種數(shù)據(jù)庫,SpringData也是Spring中與SpringBoot、SpringCloud 等齊名的知名項目,下面看下SpringBoot Mybatis Druid數(shù)據(jù)訪問的詳細(xì)過程,感興趣的朋友一起看看吧
    2021-11-11
  • IntelliJ IDEA多屏后窗口不顯示問題解決方案

    IntelliJ IDEA多屏后窗口不顯示問題解決方案

    這篇文章主要介紹了IntelliJ IDEA多屏后窗口不顯示問題解決方案,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-09-09
  • JAVA JDK8 List分組的實現(xiàn)和用法

    JAVA JDK8 List分組的實現(xiàn)和用法

    今天小編就為大家分享一篇關(guān)于JAVA JDK8 List分組的實現(xiàn)和用法,小編覺得內(nèi)容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-12-12
  • SpringBoot實現(xiàn)異步消息處理的代碼示例

    SpringBoot實現(xiàn)異步消息處理的代碼示例

    在現(xiàn)代應(yīng)用程序中,異步消息處理是一項至關(guān)重要的任務(wù)。它可以提高應(yīng)用程序的性能、可伸縮性和可靠性,同時也可以提供更好的用戶體驗,本文將介紹如何使用Spring Boot實現(xiàn)異步消息處理,并提供相應(yīng)的代碼示例
    2023-06-06
  • Springboot+Netty+Websocket實現(xiàn)消息推送實例

    Springboot+Netty+Websocket實現(xiàn)消息推送實例

    這篇文章主要介紹了Springboot+Netty+Websocket實現(xiàn)消息推送實例,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-02-02
  • Java設(shè)計模式之抽象工廠模式AbstractFactoryPattern詳解

    Java設(shè)計模式之抽象工廠模式AbstractFactoryPattern詳解

    這篇文章主要介紹了Java設(shè)計模式之抽象工廠模式AbstractFactoryPattern詳解,抽象工廠模式是一種軟件開發(fā)設(shè)計模式,抽象工廠模式提供了一種方式,可以將一組具有同一主題的單獨(dú)的工廠封裝起來,需要的朋友可以參考下
    2023-10-10

最新評論