Guava Cache示例

一、创建Employee员工类

package cn.codeduck.guavatest.pojo;

import com.google.common.base.MoreObjects;

public class Employee {
    private String name;
    private String dept;
    private String empID;

    public Employee() {
    }

    public Employee(String name, String dept, String empID) {
        this.name = name;
        this.dept = dept;
        this.empID = empID;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getDept() {
        return dept;
    }

    public void setDept(String dept) {
        this.dept = dept;
    }

    public String getEmpID() {
        return empID;
    }

    public void setEmpID(String empID) {
        this.empID = empID;
    }

    @Override
    public String toString() {
        return MoreObjects.toStringHelper(Employee.class)
                .add("name",name)
                .add("dept",dept)
                .add("empID", empID).toString();
    }
}

二、Guava Cache使用示例:

package cn.codeduck.guavatest.basic;

import cn.codeduck.guavatest.pojo.Employee;
import com.google.common.cache.*;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import com.google.common.collect.Maps;
import com.google.common.util.concurrent.ListenableFuture;
import org.junit.Test;

import java.util.List;
import java.util.Map;
import java.util.concurrent.*;

public class LoadingCacheTest {
    
    @Test
    public void test(){
        //创建一个员工LoadingCache对象
        LoadingCache<String, Employee> employeeLoadingCache = CacheBuilder.newBuilder()
                .maximumSize(100)                                    //基于容量回收:总数量100个
//                .maximumWeight(1000)                                  //基于容量回收:总权重1000
//                .weigher(new Weigher<String, Employee>() {           //配置权重
//                    @Override
//                    public int weigh(String key, Employee employee) {
//                        return key.length();
//                    }
//                })
                .expireAfterAccess(30, TimeUnit.MINUTES)      //定时回收:没有读写访问30分钟后失效清理
                .expireAfterWrite(30, TimeUnit.MINUTES)       //定时回收:没有写访问30分钟后失效清理
                .refreshAfterWrite(3, TimeUnit.SECONDS)       //定时刷新缓存,和expireAfterWrite使用,只有检索时刷新缓存。
                .weakKeys()      //基于引用回收:弱引用key
                .weakValues()    //基于引用回收:弱引用value
//                .softValues()    //基于引用回收:软引用value
                .removalListener(RemovalListeners.asynchronous(new RemovalListener<String, Employee>() {     //异步移除监听器
                    @Override
                    public void onRemoval(RemovalNotification<String, Employee> removalNotification) {
                        System.out.println("removalNotification = " + removalNotification);
                    }
                }, new Executor() {
                    @Override
                    public void execute(Runnable command) {
                        command.run();
                    }
                }))
                .recordStats()   //开启Guava Cache 统计功能
                .build(CacheLoader.asyncReloading(new CacheLoader<String, Employee>() {      //异步缓存加载器
                    // get 缓存获取不到值时候加载
                    @Override
                    public Employee load(String key) throws Exception {
                        //数据库中加载
                        return getFromDatabase(key);
                    }

                    // refresh 扩展刷新时的行为
                    @Override
                    public ListenableFuture<Employee> reload(String key, Employee oldValue) throws Exception {
                        return super.reload(key, oldValue);
                    }

                    // getAll 缓存批量获取不到值时候加载
                    @Override
                    public Map<String, Employee> loadAll(Iterable<? extends String> keys) throws Exception {
                        return super.loadAll(keys);
                    }
                }, new Executor() {
                    @Override
                    public void execute(Runnable command) {
                        command.run();
                    }
                }));

        // 1.单个查询
        System.out.println("1.单个查询");
        try {
            System.out.println("#第一次调用:");
            // 在第一次调用时,缓存将从数据库中填充相应的员工记录
            System.out.println(employeeLoadingCache.get("100"));
            System.out.println(employeeLoadingCache.get("103"));
            System.out.println(employeeLoadingCache.get("110"));
            System.out.println("#第二次调用:");
            // 在第二次调用时,会直接从缓存中读取
            System.out.println(employeeLoadingCache.get("100"));
            System.out.println(employeeLoadingCache.get("103"));
            System.out.println(employeeLoadingCache.get("110"));
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("#第三次调用:");
        // 第三次调用时 getUnchecked 不执行检查
        System.out.println(employeeLoadingCache.getUnchecked("100"));
        System.out.println(employeeLoadingCache.getUnchecked("103"));
        System.out.println(employeeLoadingCache.getUnchecked("110"));

        //1.单个查询
        //#第一次调用:
        //数据库命中100
        //Employee{name=Mahesh, dept=Finance, empID=100}
        //数据库命中103
        //Employee{name=Rohan, dept=IT, empID=103}
        //数据库命中110
        //Employee{name=Sohan, dept=Admin, empID=110}
        //#第二次调用:
        //Employee{name=Mahesh, dept=Finance, empID=100}
        //Employee{name=Rohan, dept=IT, empID=103}
        //Employee{name=Sohan, dept=Admin, empID=110}
        //#第三次调用:
        //Employee{name=Mahesh, dept=Finance, empID=100}
        //Employee{name=Rohan, dept=IT, empID=103}
        //Employee{name=Sohan, dept=Admin, empID=110}

        // 2.批量查询
        System.out.println("\n2.批量查询");
        try {
            List<String> list = Lists.newArrayList("100", "103", "110");
            ImmutableMap<String, Employee> cacheAll = employeeLoadingCache.getAll(list);
            cacheAll.forEach((k,v)->{
                System.out.println("key: "+k+", value: "+v);
            });
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        //2.批量查询
        //key: 100, value: Employee{name=Mahesh, dept=Finance, empID=100}
        //key: 103, value: Employee{name=Rohan, dept=IT, empID=103}
        //key: 110, value: Employee{name=Sohan, dept=Admin, empID=110}

        // 3.回调查询
        System.out.println("\n3.回调查询");
        try {
            Object employee = employeeLoadingCache.get("001", new Callable<Employee>() {
                @Override
                public Employee call() throws Exception {
                    return getFromDatabase("001");
                }
            });
            System.out.println(employee);
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        //3.回调查询
        //数据库命中001
        //Employee{name=Tingfeng, dept=Admin, empID=001}

        // 4.显示插入
        System.out.println("\n4.显示插入");
        employeeLoadingCache.put("001", new Employee("若风","总裁部","001"));
        System.out.println(employeeLoadingCache.getUnchecked("001"));
        //4.显示插入
        //removalNotification = 001=Employee{name=Tingfeng, dept=Admin, empID=001}
        //Employee{name=若风, dept=总裁部, empID=001}

        // 5.更改视图
        System.out.println("\n5.更改视图");
        try {
            ConcurrentMap<String, Employee> asMap = employeeLoadingCache.asMap();
            asMap.put("100",new Employee("codeduck","CTO","100"));
            ImmutableMap<String, Employee> cacheAll = employeeLoadingCache.getAll(Lists.newArrayList("100", "103", "110"));
            cacheAll.forEach((k,v)->{
                System.out.println("key: "+k+", value:"+v);
            });
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        //5.更改视图
        //removalNotification = 100=Employee{name=Mahesh, dept=Finance, empID=100}
        //key: 100, value:Employee{name=codeduck, dept=CTO, empID=100}
        //key: 103, value:Employee{name=Rohan, dept=IT, empID=103}
        //key: 110, value:Employee{name=Sohan, dept=Admin, empID=110}


        // 6.清除缓存
        System.out.println("\n6.清除所有缓存,调用移除监听器");
        employeeLoadingCache.invalidateAll();
        //6.清除所有缓存,调用移除监听器
        //removalNotification = 100=Employee{name=codeduck, dept=CTO, empID=100}
        //removalNotification = 001=Employee{name=若风, dept=总裁部, empID=001}
        //removalNotification = 110=Employee{name=Sohan, dept=Admin, empID=110}
        //removalNotification = 103=Employee{name=Rohan, dept=IT, empID=103}
    }

    // 模拟数据库中数据
    private static Employee getFromDatabase(String empId) {
        Map<String, Employee> database = Maps.newHashMap();
        database.put("100", new Employee("Mahesh", "Finance", "100"));
        database.put("103", new Employee("Rohan", "IT", "103"));
        database.put("110", new Employee("Sohan", "Admin", "110"));

        database.put("001", new Employee("Tingfeng", "Admin", "001"));
        System.out.println("数据库命中" + empId);
        return database.get(empId);
    }

}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容