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

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

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

前言

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

環(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 超時(shí)時(shí)間
   * @param unit 時(shí)間單位
   * @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è)置時(shí)間對(duì)象沒(méi)有被讀/寫訪問(wèn)則對(duì)象從內(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 長(zhǎng)度
   */
  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());
  }
}

代碼說(shuō)明

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

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

3、main演示方法中,先往緩存存入10個(gè)數(shù)據(jù),然后過(guò)5秒后每秒取一個(gè)數(shù)據(jù),并且打印一下緩存中的全部?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)不存在對(duì)應(yīng)的值了。

總結(jié)

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

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

相關(guān)文章

  • Java不可變類機(jī)制淺析

    Java不可變類機(jī)制淺析

    所謂的不可變類是指這個(gè)類的實(shí)例一旦創(chuàng)建完成后,就不能改變其成員變量值。如JDK內(nèi)部自帶的很多不可變類:Interger、Long和String等。接下來(lái)通過(guò)本文給大家介紹Java不可變類機(jī)制,需要的朋友參考下
    2017-02-02
  • Java?easyExcel的復(fù)雜表頭多級(jí)表頭導(dǎo)入

    Java?easyExcel的復(fù)雜表頭多級(jí)表頭導(dǎo)入

    最近在項(xiàng)目開發(fā)中遇到的一個(gè)excel復(fù)雜表頭的導(dǎo)入數(shù)據(jù)庫(kù)操作,下面這篇文章主要給大家介紹了關(guān)于Java?easyExcel的復(fù)雜表頭多級(jí)表頭導(dǎo)入的相關(guān)資料,需要的朋友可以參考下
    2022-06-06
  • 詳談Java泛型中T和問(wèn)號(hào)(通配符)的區(qū)別

    詳談Java泛型中T和問(wèn)號(hào)(通配符)的區(qū)別

    下面小編就為大家?guī)?lái)一篇詳談Java泛型中T和問(wèn)號(hào)(通配符)的區(qū)別。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-10-10
  • 簡(jiǎn)單了解mybatis攔截器實(shí)現(xiàn)原理及實(shí)例

    簡(jiǎn)單了解mybatis攔截器實(shí)現(xiàn)原理及實(shí)例

    這篇文章主要介紹了簡(jiǎn)單了解mybatis攔截器實(shí)現(xiàn)原理及實(shí)例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-01-01
  • SpringCloud Admin實(shí)戰(zhàn)之健康檢查與全鏈路告警深度解讀

    SpringCloud Admin實(shí)戰(zhàn)之健康檢查與全鏈路告警深度解讀

    這篇文章主要介紹了SpringCloud Admin實(shí)戰(zhàn)之健康檢查與全鏈路告警,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2025-05-05
  • 基于Spring Security前后端分離的權(quán)限控制系統(tǒng)問(wèn)題

    基于Spring Security前后端分離的權(quán)限控制系統(tǒng)問(wèn)題

    本文給大家分享基于Spring Security前后端分離的權(quán)限控制系統(tǒng)問(wèn)題,需要了解權(quán)限如何加載,權(quán)限匹配規(guī)則和登錄的實(shí)現(xiàn)代碼,對(duì)Spring Security權(quán)限控制系統(tǒng)相關(guān)知識(shí)感興趣的朋友一起看看吧
    2021-06-06
  • Java實(shí)現(xiàn)郵箱找回密碼實(shí)例代碼

    Java實(shí)現(xiàn)郵箱找回密碼實(shí)例代碼

    本篇文章主要介紹了Java實(shí)現(xiàn)郵箱找回密碼實(shí)例代碼,可以通過(guò)郵箱找回丟失密碼,具有一定的參考價(jià)值,有需要的可以了解一下。
    2016-11-11
  • 深入理解Java 類加載全過(guò)程

    深入理解Java 類加載全過(guò)程

    這篇文章主要介紹了深入理解Java 類加載全過(guò)程的相關(guān)資料,需要的朋友可以參考下
    2017-02-02
  • SpringBoot預(yù)防XSS攻擊的實(shí)現(xiàn)

    SpringBoot預(yù)防XSS攻擊的實(shí)現(xiàn)

    XSS攻擊是一種在web應(yīng)用中的計(jì)算機(jī)安全漏洞,它允許惡意web用戶將代碼植入到提供給其它用戶使用的頁(yè)面,本文主要介紹了SpringBoot預(yù)防XSS攻擊的實(shí)現(xiàn),感興趣的可以了解一下
    2023-08-08
  • Java超詳細(xì)講解排序二叉樹

    Java超詳細(xì)講解排序二叉樹

    排序二叉樹的特點(diǎn)是一個(gè)父節(jié)點(diǎn)只能有左右兩個(gè)子節(jié)點(diǎn)、左節(jié)點(diǎn)的值比父節(jié)點(diǎn)要小、右節(jié)點(diǎn)的值要比父節(jié)點(diǎn)要大,難度并不大,但是得花時(shí)間來(lái)理解
    2022-06-06

最新評(píng)論