频道栏目
首页 > 资讯 > 微信公众平台开发 > 正文

微信开发-定时获取token,保证线程安全,高可用

17-05-22        来源:[db:作者]  
收藏   我要投稿

1.微信规定token周期为7200,所以在2小时内需要再次获取,那么问题来了,假如线程程序更新周期为7100,在7100前一秒来了1000个请求(假设),这1000个请求在这一时刻获取了token(老的),又假设处理1000个请求所需时间大于1秒,定时获取token线程去更新token,这时1000个请求还未处理完的请求token就全部失效了,想要表达的就是你处理请求的过程中token发生变化,而请求中的token还没有更新。

1.1解决思路,在处理请求时加锁,在未处理完毕请求时,不会更新token。更新token时不会处理请求,一直阻塞到更新token完成,才会去处理请求。考虑阻塞时间可以自己设定token的更新周期,保证token可用。

1.1.1 可用用线程synchronize,但是这样就会有效率问题,获取token时和更新token加synchronize,问题时多线程获取token是一个一个执行的,降低了多线程的优势。

1.1.2 考虑用读写锁,这样多线程可以同时读取token,读取token时不可以更新token。更新token时,不可以获取token。

2创建读写锁,为token加锁

public class TokenWRLock {

//微信token信息,当超时对其加读写,保证微信业务正常

private static String accessToken = null;

//创建一个读写锁

private static ReadWriteLock lock = new ReentrantReadWriteLock();

/*

*

*/

public static void setToken(String token) {

try {

lock.writeLock().lock();

//Thread.sleep(1000);

accessToken = token;

} catch (Exception e) {

e.printStackTrace();

} finally {

lock.writeLock().unlock();

}

}

/*

* 获取token加锁

*/

public static String getTokenWithLock() throws Exception {

lock.readLock().lock();

//System.out.println("@@@@@"+new Date().getSeconds());

return accessToken;

//fun();

//lock.readLock().unlock();

}

/*

* 释放token锁

*/

public static void unLockToken() throws Exception {

//lock.readLock().lock();

//System.out.println("@@@@@"+new Date().getSeconds());

//fun();

lock.readLock().unlock();

}}

上面用到ReadWriteLock,这是个好东西。相比sync,它的优势就是多线程执行,不会排队。

3.创建定时获取token线程

public class TokenScheduleTask extends Thread {

Logger logger = Logger.getLogger(NotifyScheduleTask.class);

public void initMethod() {

this.start();

}

@Override

public void run() {

try {

while(true) {

//获取token

this.tokenTask(0);

Thread.sleep(7100 * 1000);

}

} catch (InterruptedException e) {

e.printStackTrace();

}

}

//定时获取token

public void tokenTask(int retry) throws InterruptedException {

try {

if(retry >= 3) {

logger.error("获取token异常重试次数以达上线");

return;

}

String token = WeiChatUtils.getAccessToken().getAccessToken();

TokenWRLock.setToken(token);

} catch (Exception e) {

retry++;

logger.error("获取token异常:",e);

Thread.sleep(1*1000);

this.tokenTask(retry);

}

}

}

收工

相关TAG标签
上一篇:用poi做excel自定义格式的导出
下一篇:L1-019. 谁先倒
相关文章
图文推荐

关于我们 | 联系我们 | 广告服务 | 投资合作 | 版权申明 | 在线帮助 | 网站地图 | 作品发布 | Vip技术培训 | 举报中心

版权所有: 红黑联盟--致力于做实用的IT技术学习网站