电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

輕松了解java中Caffeine高性能緩存庫

瀏覽:6日期:2023-12-04 14:43:52
目錄輕松lCaffeine1、依賴2、寫入緩存 2.1、手動寫入2.2、同步加載2.3、異步加載3、緩存值的清理3.1、基于大小的清理3.2、基于時間的清理 3.3、基于引用的清理4、緩存刷新5、統計輕松lCaffeine1、依賴

我們需要將Caffeine依賴添加到我們的pom.xml中:

<dependency> <groupId>com.github.ben-manes.caffeine</groupId> <artifactId>caffeine</artifactId> <version>2.5.5</version></dependency>2、寫入緩存

讓我們關注Caffeine的三種緩存寫入策略:手動、同步加載和異步加載。

首先,讓我們編寫一個類,作為要存儲在緩存中的值的類型:

class DataObject { private final String data; private static int objectCounter = 0; // standard constructors/getterspublic static DataObject get(String data) {objectCounter++;return new DataObject(data); }} 2.1、手動寫入

在此策略中,我們手動將值寫入緩存并稍后讀取它們。

我們先初始化緩存:

Cache<String, DataObject> cache = Caffeine.newBuilder() .expireAfterWrite(1, TimeUnit.MINUTES) .maximumSize(100) .build();

現在,我們可以使用getIfPresent方法從緩存中獲取一些值。如果緩存中不存在該值,則此方法將返回null:

我們可以使用put方法手動寫入緩存:

cache.put(key, dataObject);dataObject = cache.getIfPresent(key);assertNotNull(dataObject);

我們還可以使用get方法獲取值,該方法接受一個函數和一個鍵作為參數。如果緩存中不存在該鍵,則此函數將用于提供兜底值,該值將在執行后寫入緩存:

dataObject = cache .get(key, k -> DataObject.get('Data for A'));assertNotNull(dataObject);assertEquals('Data for A', dataObject.getData());

這個GET方法執行是原子性的。這意味著即使多個線程同時請求該值,執行只會進行一次。這就是為什么使用get比getIfPresent更好。

有時我們需要手動使一些緩存的值失效:

cache.invalidate(key);dataObject = cache.getIfPresent(key);assertNull(dataObject);2.2、同步加載

這種加載緩存的方法需要一個Function,用于初始化寫入值,類似于手動寫入策略的get方法,讓我們看看如何使用它。

首先,我們需要初始化我們的緩存:

現在我們可以使用get方法讀取值:

DataObject dataObject = cache.get(key);assertNotNull(dataObject);assertEquals('Data for ' + key, dataObject.getData());

我們還可以使用getAll方法獲取一組值:

Map<String, DataObject> dataObjectMap = cache.getAll(Arrays.asList('A', 'B', 'C'));assertEquals(3, dataObjectMap.size());

值從傳遞給build方法的底層后端初始化Function中讀取到,這樣就可以使用緩存作為訪問值的主要入口了。

2.3、異步加載

此策略的工作原理與前一個相同,但是會異步執行操作并返回一個CompletableFuture來保存實際的值:

AsyncLoadingCache<String, DataObject> cache = Caffeine.newBuilder() .maximumSize(100) .expireAfterWrite(1, TimeUnit.MINUTES) .buildAsync(k -> DataObject.get('Data for ' + k));

我們可以以相同的方式使用get和getAll方法,考慮到它們的返回是CompletableFuture:

String key = 'A';cache.get(key).thenAccept(dataObject -> { assertNotNull(dataObject); assertEquals('Data for ' + key, dataObject.getData());});cache.getAll(Arrays.asList('A', 'B', 'C')) .thenAccept(dataObjectMap -> assertEquals(3, dataObjectMap.size()));

CompletableFuture具有很多有用的API,您可以在本文中閱讀更多相關信息。

3、緩存值的清理

Caffeine有三種緩存值的清理策略:基于大小、基于時間和基于引用。

3.1、基于大小的清理

這種類型的清理設計為在超出緩存配置的大小限制時發生清理。有兩種獲取大小的方法——計算緩存中的對象數,或者獲取它們的權重。

讓我們看看如何計算緩存中的對象數。緩存初始化時,其大小為零:

LoadingCache<String, DataObject> cache = Caffeine.newBuilder() .maximumSize(1) .build(k -> DataObject.get('Data for ' + k));assertEquals(0, cache.estimatedSize());

當我們添加一個值時,大小明顯增加:

cache.get('A');assertEquals(1, cache.estimatedSize());

我們可以將第二個值添加到緩存中,這會導致刪除第一個值:

cache.get('B');cache.cleanUp();assertEquals(1, cache.estimatedSize());

值得一提的是,我們在獲取緩存大小之前調用了cleanUp方法。這是因為緩存清理是異步執行的,該方法有助于等待清理完成。

我們還可以傳入一個weigher的Function來定義緩存大小的獲取:

LoadingCache<String, DataObject> cache = Caffeine.newBuilder() .maximumWeight(10) .weigher((k,v) -> 5) .build(k -> DataObject.get('Data for ' + k));assertEquals(0, cache.estimatedSize());cache.get('A');assertEquals(1, cache.estimatedSize());cache.get('B');assertEquals(2, cache.estimatedSize());

當權重超過 10 時,這些值將從緩存中刪除:

cache.get('C');cache.cleanUp();assertEquals(2, cache.estimatedSize());3.2、基于時間的清理

這種清理策略基于條目的過期時間,分為三種:

訪問后過期——自上次讀取或寫入以來,條目在經過某段時間后過期寫入后過期——自上次寫入以來,條目在經過某段時間后過期自定義策略——由Expiry的實現來為每個條目單獨計算到期時間讓我們使用expireAfterAccess方法配置訪問后過期策略:

LoadingCache<String, DataObject> cache = Caffeine.newBuilder() .expireAfterAccess(5, TimeUnit.MINUTES) .build(k -> DataObject.get('Data for ' + k));

要配置寫入后過期策略,我們使用expireAfterWrite方法:

cache = Caffeine.newBuilder() .expireAfterWrite(10, TimeUnit.SECONDS) .weakKeys() .weakValues() .build(k -> DataObject.get('Data for ' + k));

要初始化自定義策略,我們需要實現Expiry接口:

cache = Caffeine.newBuilder().expireAfter(new Expiry<String, DataObject>() { @Override public long expireAfterCreate( String key, DataObject value, long currentTime) {return value.getData().length() * 1000; } @Override public long expireAfterUpdate( String key, DataObject value, long currentTime, long currentDuration) {return currentDuration; } @Override public long expireAfterRead( String key, DataObject value, long currentTime, long currentDuration) {return currentDuration; }}).build(k -> DataObject.get('Data for ' + k)); 3.3、基于引用的清理

我們可以配置我們的緩存,允許緩存的鍵或值或二者一起的垃圾收集。為此,我們需要為鍵和值配置WeakReference的使用,并且我們可以配置SoftReference僅用于值的垃圾收集。

WeakReference的使用允許在沒有對對象的任何強引用時對對象進行垃圾回收。SoftReference允許基于JVM的全局LRU(最近最少使用)策略對對象進行垃圾回收。可以在此處找到有關Java中引用的更多詳細信息。

我們使用Caffeine.weakKeys()、Caffeine.weakValues()和Caffeine.softValues()來啟用每個選項:

LoadingCache<String, DataObject> cache = Caffeine.newBuilder() .expireAfterWrite(10, TimeUnit.SECONDS) .weakKeys() .weakValues() .build(k -> DataObject.get('Data for ' + k));cache = Caffeine.newBuilder() .expireAfterWrite(10, TimeUnit.SECONDS) .softValues() .build(k -> DataObject.get('Data for ' + k));4、緩存刷新

可以將緩存配置為在定義的時間段后自動刷新條目。讓我們看看如何使用refreshAfterWrite方法做到這一點:

Caffeine.newBuilder() .refreshAfterWrite(1, TimeUnit.MINUTES) .build(k -> DataObject.get('Data for ' + k));

在這里,我們應該明白expireAfter和refreshAfter的一個區別:當請求過期條目時,執行會阻塞,直到build函數計算出新值。但是如果該條目符合刷新條件,則緩存將返回一個舊值并異步重新加載該值。

5、統計

Caffeine提供了一種記錄緩存使用統計信息的方法:

LoadingCache<String, DataObject> cache = Caffeine.newBuilder() .maximumSize(100) .recordStats() .build(k -> DataObject.get('Data for ' + k));cache.get('A');cache.get('A');assertEquals(1, cache.stats().hitCount());assertEquals(1, cache.stats().missCount());

到此這篇關于輕松了解java中Caffeine高性能緩存庫的文章就介紹到這了,更多相關java Caffeine緩存庫內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
主站蜘蛛池模板: 合金耐磨锤头_破碎机锤头_郑州市德勤建材有限公司 | 仿真茅草_人造茅草瓦价格_仿真茅草厂家_仿真茅草供应-深圳市科佰工贸有限公司 | 科昊仪器超纯水机系统-可成气相液氮罐-美菱超低温冰箱-西安昊兴生物科技有限公司 | 礼仪庆典公司,礼仪策划公司,庆典公司,演出公司,演艺公司,年会酒会,生日寿宴,动工仪式,开工仪式,奠基典礼,商务会议,竣工落成,乔迁揭牌,签约启动-东莞市开门红文化传媒有限公司 | 品牌策划-品牌设计-济南之式传媒广告有限公司官网-提供品牌整合丨影视创意丨公关活动丨数字营销丨自媒体运营丨数字营销 | 浙江皓格药业有限公司| 蓝米云-专注于高性价比香港/美国VPS云服务器及海外公益型免费虚拟主机 | 北京模型公司-军事模型-工业模型制作-北京百艺模型沙盘公司 | 深圳工程师职称评定条件及流程_深圳职称评审_职称评审-职称网 | 山东臭氧发生器,臭氧发生器厂家-山东瑞华环保设备 | 板式换热器_板式换热器价格_管式换热器厂家-青岛康景辉 | wika威卡压力表-wika压力变送器-德国wika代理-威卡总代-北京博朗宁科技 | 烟台条码打印机_烟台条码扫描器_烟台碳带_烟台数据采集终端_烟台斑马打印机-金鹏电子-金鹏电子 | 酒精检测棒,数显温湿度计,酒安酒精测试仪,酒精检测仪,呼气式酒精检测仪-郑州欧诺仪器有限公司 | 电梯乘运质量测试仪_电梯安全评估测试仪-武汉懿之刻 | 示波器高压差分探头-国产电流探头厂家-南京桑润斯电子科技有限公司 | 福州甲醛检测-福建室内空气检测_环境检测_水质检测-福建中凯检测技术有限公司 | 北京开业庆典策划-年会活动策划公司-舞龙舞狮团大鼓表演-北京盛乾龙狮鼓乐礼仪庆典策划公司 | 剪刃_纵剪机刀片_分条机刀片-南京雷德机械有限公司 | 胀套-锁紧盘-风电锁紧盘-蛇形联轴器「厂家」-瑞安市宝德隆机械配件有限公司 | 浴室柜-浴室镜厂家-YINAISI · 意大利设计师品牌 | 咿耐斯 |-浙江台州市丰源卫浴有限公司 | 工业车间焊接-整体|集中除尘设备-激光|等离子切割机配套除尘-粉尘烟尘净化治理厂家-山东美蓝环保科技有限公司 | 示波器高压差分探头-国产电流探头厂家-南京桑润斯电子科技有限公司 | 喷码机,激光喷码打码机,鸡蛋打码机,手持打码机,自动喷码机,一物一码防伪溯源-恒欣瑞达有限公司 | 品牌设计_VI设计_电影海报设计_包装设计_LOGO设计-Bacross新越品牌顾问 | 合肥汽车充电桩_安徽充电桩_电动交流充电桩厂家_安徽科帝新能源科技有限公司 | 华东师范大学在职研究生招生网_在职研究生招生联展网 | 圈酒招商网【jiushuitv.com】_酒水招商_代理_加盟平台 | 广西教师资格网-广西教师资格证考试网 | 杭州画室_十大画室_白墙画室_杭州美术培训_国美附中培训_附中考前培训_升学率高的画室_美术中考集训美术高考集训基地 | 钢格板|镀锌钢格板|热镀锌钢格板|格栅板|钢格板|钢格栅板|热浸锌钢格板|平台钢格板|镀锌钢格栅板|热镀锌钢格栅板|平台钢格栅板|不锈钢钢格栅板 - 专业钢格板厂家 | 液压油缸-液压缸厂家价格,液压站系统-山东国立液压制造有限公司 液压油缸生产厂家-山东液压站-济南捷兴液压机电设备有限公司 | 新疆散热器,新疆暖气片,新疆电锅炉,光耀暖通公司 | 小型单室真空包装机,食品单室真空包装机-百科 | 胃口福饺子加盟官网_新鲜现包饺子云吞加盟 - 【胃口福唯一官网】 | 橡胶接头_橡胶软接头_套管伸缩器_管道伸缩器厂家-巩义市远大供水材料有限公司 | 锂电混合机-新能源混合机-正极材料混料机-高镍,三元材料混料机-负极,包覆混合机-贝尔专业混合混料搅拌机械系统设备厂家 | 嘉兴泰东园林景观工程有限公司_花箱护栏 | HDPE土工膜,复合土工膜,防渗膜价格,土工膜厂家-山东新路通工程材料有限公司 | 电梯乘运质量测试仪_电梯安全评估测试仪-武汉懿之刻 | 生态板-实木生态板-生态板厂家-源木原作生态板品牌-深圳市方舟木业有限公司 |