背景:
客户端要调用微信JS_ SDK,需要后端返回jsapi_ticket给前端进行签名,而jsapi_ticket又需要通过accessToken获取。
微信公众号的accessToken是有过期时间的,文档写是2个小时,我们可以通过指定api去刷新accessToken,但是有一点是,如果刷新了accessToken,那么原来的accessToken将会变得不可用,这就可能出现用一个不可用的accessToken来进行请求返回错误。
一般的做法是使用Redis作为全局缓存,存入有过期时间的key,然后定时去刷新这个Redis以及key过期自动去刷新。一般情况下,尽管可能还是会出现多个线程过来,其中有一个拿到的是过期的accessToken的情况,但是其实这种情况很少会发生,就算发生,再请求一次即可。
但是抱着学习成长的心态,决定使用读写锁来控制accessToken的操作,在刷新accessToken的时候加上写锁,此时其他线程不能去获取accessToken,等到写锁释放的时候,一定是拿到最新的。
/**
* 从redis中获取JsapiTicket
*/
public String getJsapiTicketCacheTest(){
String xdlmJsapiTicket = null;
if(env.equals(DEV_ENV)){
// 测试环境逻辑
System.out.println("线程:" + Thread.currentThread().getName() + "测试环境获取xdlmJsapiTicket数据结果为空");
} else {
// 正式环境逻辑
System.out.println("正式环境获取xdlmJsapiTicket数据结果为空");
}
if(StringUtils.isNoneBlank(xdlmJsapiTicket)){
return xdlmJsapiTicket;
} else {
return getJsapiTicketTest(xdzOpenWechatConfig);
}
}
/**
* accessToken只能在executeTest内部调用,以支撑锁体系
* @return
*/
private String getAccessTokenCacheTest(){
// accessTokenRLock.lock(); // 获取读锁
// System.out.println("线程:" + Thread.currentThread().getName() + "----------获取读锁成功");
try {
String accessToken = null;
if(env.equals(DEV_ENV)){
// 测试环境逻辑
System.out.println("线程:" + Thread.currentThread().getName() + "测试环境获取accessToken数据结果为空");
}