A. 背景
手上有个项目,使用某产品(简称APIM)提供的 REST API 重新开发一套 web 应用。该产品的 REST API 调用时间比较长,同时该 web 应用的读操作远多于写操作,是个典型的缓存使用场景。决定在这个项目中引入 Redis,学习一下这个火了很久的技术(有点后知后觉了)。
B. 设计思路
目前已有 APIMRestClient 类负责调用 APIM 的 REST API,将返回的 Json 转化为 POJO:
public PlanVO getPublicPlan(String planId, String version) {
ResponseHandler<PlanVO> rh = new APIMResponseHandler<PlanVO>(new TypeReference<PlanVO>() {
});
return invokeAPIMRestAPIByGet("/plans/" + planId + "/v" + version, rh);
}
第一步是直接在代码中加入缓存功能片段,伪代码如下:
public PlanVO getPublicPlan(String planId, String version) {
// 拼接 key。此处为 PublicPlan.planId的值.version的值
// 从 Redis 中根据 key 获取缓存的对象
// 如果缓存对象不为空,则直接返回缓存对象
ResponseHandler<PlanVO> rh = new APIMResponseHandler<PlanVO>(new TypeReference<PlanVO>() {
});
PlanVO result = invokeAPIMRestAPIByGet("/plans/" + planId + "/v" + version, rh);
// 将结果存入 Redis 中
return result;
}
其中有个问题:如何在 Redis 中保存 POJO 对象(Java对象)?一般来说有两种思路:使用 Java 原生的序列化机制将对象转化为 byte[],或者将对象转化为 Json 字符串。考虑到这些 POJO 对象本来就是用于将 APIM REST API 返回的 Json 转化为对象而创建的,所以倾向于使用 Json 的方式来“序列化”对象。
第二步,由于 APIMRestClient 类中有许多类似的方法(对应不同的模型),所以引入 AOP 解决重复的代码逻辑。
第三步,对修改和删除操作直接修改缓存对象的状态,提高 web 应用的友好性。
C. Redis 初探
博主本人也是初次接触 Redis,于是在网上找了个视频,加速到 150% 看完。个人觉得这是学习新知识最快的入门方法了。
Redis 的知识就不细说了,网上资料多得是。只列出和本项目有关的部分,供您回忆或是快速入门。
Redis 是一个 key-value 存储系统。它支持存储的 value 类型 包括string(字符串)、list(链表)、set(集合)、zset(sorted set –有序集合)和 hash(哈希类型)。我只用到了 string 这种值类型。
Redis 本身只能运行在 Linux 系统上,不过微软提供了 Windows 可用的版本
https://github.com/MSOpenTech/redis/tree/win-2.8.17.2
以上版本需要VS编译,最好直接找个编译好的版本
Windows 版解压缩即可使用。双击 redis-server.exe 打开服务程序,双击 redis-cli.exe 打开客户端。客户端可以打开多个。
列出几个简单命令:
redis 127.0.0.1:6379> select 5 // Redis有16个数据库,编号0~15。这句命令表示选择5号数据库。
OK
redis 127.0.0.1:6379[5]> set name.gordon "Gordon Kong" // 设值,key为name.gordon
OK
redis 127.0.0.1:6379[5]> get name.gordon // 取值
"Gordon Kong"
redis 127.0.0.1:6379[5]> setnx name.ben Ben // 若key未被使用,则设值,否则返回0
(integer) 1
redis 127.0.0.1:6379[5]> get name.ben
"Ben"
redis 127.0.0.1:6379