Google的guava是个很好的项目,提供了诸如集合、缓存、并发、String工具类等等,实乃Java开发利器。这里简单说一下LoadingCache使用时,对于不存在的值如何优雅处理。ifeve.com有翻译cache相关的介绍,在这里

使用Cache时,我们优先读取缓存,当缓存不存在时,则从实际的数据存储获取,如DB、磁盘、网络等,即get-if-absent-compute。guava提供了CacheLoader机制,允许我们通过设置Loader来自动完成这一过程。如:

1
Cache<String, User> cache = CacheBuilder.newBuilder().expireAfterAccess(5, TimeUnit.MINUTES);
2
user = cache.get(name, () -> {
3
    User value = query(key);//from databse, disk, etc.
4
    return value;
5
});

或者使用LoadingCache:

1
 LoadingCache<String, User> cache = CacheBuilder.newBuilder().expireAfterAccess(5, TimeUnit.MINUTES).build(
2
     new CacheLoader<String, User>() {
3
        @Override
4
        public User load(String name) throws Exception {
5
        User value = query(key);//from databse, disk, etc.
6
        return value;
7
        }
8
    }
9
);

这可比自己写一个Map来缓存数据方便多了,而且还可以设置超时时间自动帮我们清理过期的数据。

不过需要注意一点的是,CacheLoader不允许返回的数据为NULL,否则会抛出异常:CacheLoader returned null for key。所以我们需要保证查找的数据必须存在,或者抛出异常外部处理。在某些情况下,我们的数据可能确实不在,比如用户管理模块,我们在新增数据前,要查询原来是否已经存在该用户,那么这时候抛出异常也不合适,此时可以使用Optional来优化CacheLoader:

1
 LoadingCache<String, Optional<User>> cache = CacheBuilder.newBuilder().expireAfterAccess(5, TimeUnit.MINUTES).build(
2
     new CacheLoader<String, Optional<User>>() {
3
        @Override
4
        public Optional<User> load(String name) throws Exception {
5
        User value = query(key);//from databse, disk, etc.
6
        return Optional.ofNullable(value);
7
        }
8
    }
9
);

这样我们保证了CacheLoader返回值不为NULL,而业务数据是否存在,只需要判断Optional.ifPresent()就行了,同时Optional的其他函数在业务逻辑中也是非常有用的。