您好,欢迎来到意题教育。
搜索
您的当前位置:首页redis实现高并发计数

redis实现高并发计数

来源:意题教育


业务需求中经常有需要用到计数器的场景:譬如一个手机号一天发送5条短信、一个接口一分钟多少请求、一个接口一天调用多少次等等。使用Redis的Incr自增命令可以轻松实现以上需求。以一个接口一天调用次数为例:

	/**
	 * 是否拒绝服务
	 * @return
	 */
	private boolean denialOfService(String userId){
	long count=JedisUtil.setIncr(DateUtil.getDate()+"&"+userId+"&"+"queryCarViolation", 800);
	if(count<=10){
	return false;
	}
	return true;
	}
 /**
	 * 查询违章
	 * @param plateNumber车牌
	 * @param vin 车架号
	 * @param engineNo发动机
	 * @param request
	 * @param response
	 * @throws Exception
	 */
	@RequestMapping("/queryCarViolationList.json")
	@AuthorizationApi
	public void queryCarViolationList(@CurrentToken Token token,String plateNumber,String vin,
 String engineNo,HttpServletRequest request,HttpServletResponse response) throws Exception {
	 String userId=token.getUserId();
 //超过,拦截请求
 if(denialOfService(userId)){
	 apiData(request, response, ReqJson.error(CarError.ONLY_5_TIMES_A_DAY_CAN_BE_FOUND));
	 return;
	 }
	//没超过,业务逻辑……
 }

每次调用接口之前,先获得下计数器自增后的值,如果小于,放行,执行后面的代码。如果大于,则拦截掉。

JedisUtil工具类:

public class JedisUtil {
	protected final static Logger logger = Logger.getLogger(JedisUtil.class);
	private static JedisPool jedisPool;
	
	@Autowired(required = true)
	public void setJedisPool(JedisPool jedisPool) {
	JedisUtil.jedisPool = jedisPool;
	}
	/**
	 * 对某个键的值自增
	 * @author liboyi
	 * @param key 键
	 * @param cacheSeconds 超时时间,0为不超时
	 * @return
	 */
	public static long setIncr(String key, int cacheSeconds) {
	long result = 0;
	Jedis jedis = null;
	try {
	jedis = jedisPool.getResource();
	result =jedis.incr(key);
	if (cacheSeconds != 0) {
	 jedis.expire(key, cacheSeconds);
	}
	logger.debug("set "+ key + " = " + result);
	} catch (Exception e) {
	logger.warn("set "+ key + " = " + result);
	} finally {
	jedisPool.returnResource(jedis);
	}
	return result;
	}
}	

更多redis知识请关注redis入门教程栏目。

Copyright © 2019- itiz.cn 版权所有

违法及侵权请联系:TEL:199 18 7713 E-MAIL:2724546146@qq.com

本站由北京市万商天勤律师事务所王兴未律师提供法律服务