ExpiringMap,一个可以用于缓存的 Map

2020-09-28 14:41:10 浏览数 (2)

1.1 简介

1.1.1 概述

  ExpiringMap 是一个高性能,低开销,零依赖,由线程安全的 ConcurrentMap 实现可以设置过期时间的 Map。功能包括:可设置 Map 中的 Entry 在一段时间后自动过期。可设置 Map 最大容纳值,当到达 Maximum size 后,再次插入值会导致 Map 中的第一个值过期。可添加监听事件,在监听到 Entry 过期时调度监听函数。可以设置懒加载,在调用 get() 方法时创建对象。☛ 项目地址

1.1.2 常用 API

  ♞ enum ExpirationPolicy:枚举,确定 ExpiringMap 元素应如何过期,ACCESSED 根据上次访问的时间使它们过期,即从上次访问后开始计时;CREATED 根据条目的创建时间使它们过期,即从创建之后开始计时。   ♞ enum TimeUnit:枚举,确定时间单位,MILLISECONDS 时间单位毫秒,SECONDS 时间单位秒,MINUTES 时间单位分钟,HOURS 时间单位小时,DAYS 时间单位天   ♞ interface EntryLoader<K, V>:按需加载,内有 V load(K key) 方法,调用时将 key 的新值加载到即将过期的映射中。   ♞ interface ExpiringEntryLoader<K, V>:按需加载,并控制每个值的过期时间(即可变过期时间),内有 ExpiringValue<V> load(K key) 方法,调用时将 key 的新值加载到即将过期的映射中。   ♞ static final class Builder<K, V>:内部类,ExpiringMap 构造器。 默认为 ExpirationPolicy.CREATED,60 个 TimeUnit.SECONDS 到期和 Integer.MAX_VALUE 的 MAXSIZE。   ♞ static Builder<Object, Object> builder():创建一个 ExpiringMap 构造器,返回 ExpiringMap.Builder。   ♞ Builder<K, V> expiration(long duration, TimeUnit timeUnit):构造器中的方法,用于设置 Map 的过期时间,返回 ExpiringMap.Builder。   ♞ Builder<K, V> maxSize(int maxSize):设置 Map 的最大个数,超过限制仍要添加元素时将最先过期的元素过期。   ♞ int getMaxSize():获取 Map 的最大个数。   ♞ Builder<K1, V1> entryLoader(EntryLoader<? super K1, ? super V1> loader):构造器中的方法,可以设置 EntryLoader,EntryLoader 和 ExpiringEntryLoader,不能同时设置。   ♞ Builder<K1, V1> expiringEntryLoader(ExpiringEntryLoader<? super K1, ? super V1> loader):构造器中的方法,可以设置 ExpiringEntryLoader,EntryLoader 和 ExpiringEntryLoader,不能同时设置。   ♞ Builder<K, V> expirationPolicy(ExpirationPolicy expirationPolicy):构造器中的方法,用于设置 Map 的过期策略,返回 ExpiringMap.Builder。   ♞ Builder<K, V> variableExpiration():构造器中的方法,允许 Map 元素具有各自的到期时间,并允许更改到期时间。   ♞ Builder<K1, V1> expirationListener(ExpirationListener<? super K1, ? super V1> listener):构造器中的方法,配置监听每个 Map 元素过期, 通知传入的 ExpirationListener 执行预定好的操作。   ♞ void addExpirationListener(ExpirationListener<K, V> listener):给 ExpiringMap 添加过期监听器。   ♞ void removeExpirationListener(ExpirationListener<K, V> listener):移出 ExpiringMap 的过期监听。   ♞ long getExpiration(K key):获取指定 key 对应的元素的过期时间,返回毫秒值。   ♞ long getExpectedExpiration(K key):获取指定 key 预计到期时间,返货毫秒值。   ♞ ExpirationPolicy getExpirationPolicy(K key):获取指定 key 对应的元素的过期策略。   ♞ interface ExpirationListener<K, V>:过期监听接口,有一个 void expired(K key, V value) 方法,过期时自动调用。   ♞ ExpiringMap<K1, V1> build():创建并返回一个 ExpiringMap。

1.1.3 Maven 依赖

代码语言:javascript复制
<dependency>
    <groupId>net.jodah</groupId>
    <artifactId>expiringmap</artifactId>
    <version>0.5.8</version>
</dependency>

1.2 示例

1.2.1 设置过期时间与过期协议

代码语言:javascript复制
/**
 * Created with IntelliJ IDEA.
 *
 * @author Demo_Null
 * @date 2020/8/6
 * @description 构建时设置过期时间与过期策略
 */
public class MyTest {

    ExpiringMap<String, String> ACCESSED = ExpiringMap.builder().expiration(10, TimeUnit.SECONDS)
            .expirationPolicy(ExpirationPolicy.ACCESSED)
            .build();

    ExpiringMap<String, String> CREATED = ExpiringMap.builder().expiration(10, TimeUnit.SECONDS)
            .expirationPolicy(ExpirationPolicy.CREATED)
            .build();

    @Test
    public void testA() throws InterruptedException {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        ACCESSED.put("name", "小张");
        CREATED.put("name", "小李");
        
        System.out.println(sdf.format(new Date()));
        System.out.println("ACCESSED name 还有 "   ACCESSED.getExpectedExpiration("name")
                  " ms过期, name = "   ACCESSED.get("name"));
        System.out.println("CREATED name 还有 "   CREATED.getExpectedExpiration("name")
                  " ms过期, name = "   CREATED.get("name"));

        Thread.sleep(6000);
        System.out.println(sdf.format(new Date()));
        System.out.println("ACCESSED name 还有 "   ACCESSED.getExpectedExpiration("name")
                  " ms过期, name = "   ACCESSED.get("name"));
        System.out.println("CREATED name 还有 "   CREATED.getExpectedExpiration("name")
                  " ms过期, name = "   CREATED.get("name"));

        Thread.sleep(5000);
        System.out.println(sdf.format(new Date()));
        System.out.println("ACCESSED name 还有 "   ACCESSED.getExpectedExpiration("name")
                  " ms过期, name = "   ACCESSED.get("name"));
        System.out.println("CREATED name 已过期, name = "   CREATED.get("name"));

        Thread.sleep(11000);
        System.out.println(sdf.format(new Date()));
        System.out.println("ACCESSED name 已过期, name = "   ACCESSED.get("name"));
        System.out.println("CREATED name 已过期, name = "   CREATED.get("name"));
    }
}

1.2.2 单个元素设置过期时间

代码语言:javascript复制
/**
 * Created with IntelliJ IDEA.
 *
 * @author Demo_Null
 * @date 2020/8/6
 * @description 构建后给单个元素设置过期时间
 */
public class MyTest {

    ExpiringMap<String, String> map = ExpiringMap.builder().variableExpiration().build();

    @Test
    public void testA() throws InterruptedException {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
        map.put("name", "小张", ExpirationPolicy.CREATED, 5000, TimeUnit.MILLISECONDS);

        System.out.println(sdf.format(new Date()));
        System.out.println("还有"   map.getExpectedExpiration("name")   " ms过期 ,name="   map.get("name"));

        Thread.sleep(5001);
        System.out.println(sdf.format(new Date()));
        System.out.println( "已过期 ,name="   map.get("name"));
    }
}

1.2.3 设置过期监听

代码语言:javascript复制
/**
 * Created with IntelliJ IDEA.
 *
 * @author Demo_Null
 * @date 2020/8/6
 * @description 构建时设置过期监听
 */
public class MyTest {

    ExpiringMap<String, String> map = ExpiringMap.builder().variableExpiration()
                    .expirationListener((thekey, thevalue) -> {
                            System.out.println("key: "   thekey   " -- value: "   thevalue   " 过期");
                    }).build();

    @Test
    public void testA() throws InterruptedException {
        map.put("name", "小张", ExpirationPolicy.CREATED, 5000, TimeUnit.MILLISECONDS);

        Thread.sleep(5001);
        System.out.println( "已过期 ,name="   map.get("name"));
    }
}

1.2.4 添加过期监听

代码语言:javascript复制
/**
 * Created with IntelliJ IDEA.
 *
 * @author Demo_Null
 * @date 2020/8/6
 * @description 构建后添加过期监听
 */
public class MyTest {

    ExpiringMap<String, String> map = ExpiringMap.builder().variableExpiration().build();

	// 可用 Lambda 表达式 (key, value) -> System.out.println()
    ExpirationListener<String, String> expirationListener = new ExpirationListener<String, String>() {
        @Override
        public void expired(String key, String value) {
            System.out.println("key: "   key   " -- value: "   value   " 过期");
        }
    };

    @Test
    public void testA() throws InterruptedException {
        map.put("name", "小张", ExpirationPolicy.CREATED, 5000, TimeUnit.MILLISECONDS);
        map.addExpirationListener(expirationListener);

        Thread.sleep(5001);
        System.out.println( "已过期 ,name="   map.get("name"));
    }
}

1.2.5 懒加载

代码语言:javascript复制
/**
 * Created with IntelliJ IDEA.
 *
 * @author Demo_Null
 * @date 2020/8/6
 * @description 懒加载
 */
public class MyTest {

    ExpiringMap<String, Student> map = ExpiringMap.builder().expiration(5000, TimeUnit.MILLISECONDS)
            .entryLoader(key -> new Student()).build();

    ExpirationListener<String, Student> expirationListener = new ExpirationListener<String, Student>() {
        @Override
        public void expired(String key, Student value) {
            System.out.println("key: "   key   " -- value: "   value   " 过期");
        }
    };

    @Test
    public void testA() throws InterruptedException {
        map.addExpirationListener(expirationListener);
		// 使用懒加载时,可以不用去 put 对象,在调用 get 方法时会自动创建对象
        Student student = map.get("student");
        student.setName("张三");
        student.setAge(23);
        System.out.println(map.get("student"));
        
        Thread.sleep(5001);
    }
}

class Student {
    private String name;
    private Integer age;

    public String getName() {
        return name;
    }

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

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    @Override
    public String toString() {
        return "Student{"  
                "name='"   name   '''  
                ", age="   age  
                '}';
    }
}

0 人点赞