Java小技巧:创建带缓存的过程

在平常开发中,我们经常遇到这样一类过程:有一定计算量,被频繁调用,但对于任意一个参数,结果是恒定的(换句话说,就是纯函数),为了减少频繁调用的性能开销,我们可能会写出这样的代码:

java 复制代码
public class ExampleUnitTest {

    Map<Integer, String> cacheMap = new HashMap<>();

    public String doTaskWithCache(Integer i) {
        String s = cacheMap.get(i); // 先检查是否有缓存
        if (s == null) {
            s = calculateResult(i);
            cacheMap.put(i, s);  // 计算结果存到缓存里
        }
        return s;
    }
}

但是,这样写的话,每次有需要带缓存的函数,创建缓存以及检查结果是否已缓存的那代码就得重写一遍,想想就有些麻烦,本文就介绍一个简单的工具Memorizer来简化这个过程。

Memorizer 的实现

因为实现代码比较少,我们看实现代码

java 复制代码
public class Memorizer<T, V> {
    public static <T, V> Function<T, V> memorize(Function<T, V> function) {
        Map<T, V> cacheMap = new HashMap<>();
        return (T t) -> Optional.ofNullable(cacheMap.get(t)).orElseGet(() -> {
            V value = function.apply(t);
            cacheMap.put(t, value);
            return value;
        });
    }

    public static <T, V> Function<T, V> weakMemorize(Function<T, V> function) {
        Map<T, V> cacheMap = new WeakHashMap<>();
        return (T t) -> Optional.ofNullable(cacheMap.get(t)).orElseGet(() -> {
            V value = function.apply(t);
            cacheMap.put(t, value);
            return value;
        });
    }
}

这个工具有两个方法,都接受一个单参函数作为参数,并返回一个同类型的单参函数,意思是:传入要缓存的函数,返回该函数的带缓存的版本。

weakMemorize方法返回的函数的缓存可能会被回收,适合那些内存资源紧张或者想减少内存压力的场景,不过这也会导致一定概率的缓存失效的情况。

Memorizer 的使用

我们用``来改写上面的例子,就成了:

java 复制代码
public class ExampleUnitTest {
    private Function<Integer, String> doTaskWithCache = (i) -> calculateResult(i);
}

写在最后

这里只给出了单参数的实现,如果要考虑多参数的缓存,可以尝试用下面这种方式扩展:

java 复制代码
public class Memorizer<T, V> {
    public static <T1, T2, V, K> BiFunction<T1, T2, V> memorize(BiFunction<T1, T2, V> function, BiFunction<T1, T2, K> keyExtractor) {
        Map<K, V> cacheMap = new HashMap<>();
        return (t1, t2) -> {
            K key = keyExtractor.apply(t1, t2);
            return Optional.ofNullable(cacheMap.get(key)).orElseGet(() -> {
                V value = function.apply(t1, t2);
                cacheMap.put(key, value);
                return value;
            });
        };
    }

    public static <T1, T2, V, K> BiFunction<T1, T2, V> weakMemorize(BiFunction<T1, T2, V> function, BiFunction<T1, T2, K> keyExtractor) {
        Map<K, V> cacheMap = new WeakHashMap<>();
        return (t1, t2) -> {
            K key = keyExtractor.apply(t1, t2);
            return Optional.ofNullable(cacheMap.get(key)).orElseGet(() -> {
                V value = function.apply(t1, t2);
                cacheMap.put(key, value);
                return value;
            });
        };
    }
}
相关推荐
计算机学姐3 分钟前
基于SpringBoot的高校体育场馆预约系统【个性化推荐算法+数据可视化统计】
java·vue.js·spring boot·后端·mysql·信息可视化·推荐算法
Coder_Boy_5 分钟前
基于SpringAI的在线考试系统设计-用户管理模块设计
java·大数据·人工智能·spring boot·spring cloud
小白不想白a11 分钟前
RabbitMQ监控
java·rabbitmq·java-rabbitmq
冬奇Lab14 分钟前
稳定性性能系列之九——启动性能优化:Boot、冷启动与热启动
android·性能优化
Overt0p16 分钟前
MQ简单介绍以及RabbitMQ基础使用,快速上手
java·分布式·rabbitmq
STCNXPARM18 分钟前
Android 显示系统 - View体系、WMS
android·wms·view·android显示子系统
奋进的芋圆18 分钟前
SerialCommManager 详解:从嵌入式通信管理器到 Spring Boot 后端服务
java·spring boot·接口隔离原则
奋进的芋圆18 分钟前
Spring Boot + RAG 项目中集成 MCP 接口技术文档
java·spring boot·ai
sxlishaobin20 分钟前
设计模式之装饰器模式
java·设计模式·装饰器模式
weixin_4469388721 分钟前
谷歌play上架广告app
android