Redis incr解决并发问题

柔情只为你懂 2021-10-15 01:25 701阅读 0赞

项目背景:

1、新增问题件工单,工单中有工单编码字段,工单编码字段的规则为 “WT”+yyyyMMdd+0000001。

2、每天的工单生成量是30W,所以会存在并发问题

解决思路:

1、首先乐观的认为redis不会宕机,对应的缓存不会被清除(除非人为操作,人为操作会有独立的补救办法)

2、将工单编码存到缓存中(redis),其值只存“WT”+yyyyMMdd后面的数字部分;

  1. 对应的key为:key标识+yyyyMMdd,即每天一个key

3、每次生成工单编码时,先调用redis的incr方法,使其在原来编码的基础上加1,并返回结果

4、判断返回的结果,如果返回的是1,说明当前key之前不存在,为生成的新的一天的key,

  1. 需要设置此key的生命周期为24小时

5、每个key只会存活24小时

6、如果redis宕机,或者key被删除,调用指定的接口,接口会去数据库查询今天最大的工单编码,

  1. 解析后,将其存在redis中,后面的工单编码再在此基础上自增

7、请自行配置redisClient客户端并实例化

代码:

1、编码获取核心方法

  1. /**
  2. * 通过自定义的方法查询问题件缓存
  3. * @param redisKey
  4. * @param codePre
  5. * @return
  6. */
  7. public static String getCodeBySelfRedis(String redisKey,String codePre){
  8. String nowDateStr = DateTimeUtil.getDateTimeStr(new Date(),DateTimeUtil.DATE_PATTERN_YYYYMMDD);
  9. Long value = 1L;
  10. RedisClient uceClient = null;
  11. Jedis jedis=null;
  12. try {
  13. uceClient = SpringContextUtil.getBean("uceClient");
  14. jedis = uceClient.getResource();
  15. value = jedis.incr(redisKey+nowDateStr);
  16. if(value != null){
  17. //如果值为1说明是第一次设置,那么设置key的存活时间为24小时
  18. if (value == 1){
  19. jedis.expire(redisKey+nowDateStr,(24*60*60+1000));
  20. }
  21. }else{
  22. //如指为空,重置缓存
  23. value = resetCodeRedis(redisKey);
  24. }
  25. }catch (Exception e){
  26. logger.error("获取问题件编码的自定义缓存异常:",e);
  27. value = resetCodeRedis(redisKey);
  28. }finally {
  29. if (uceClient != null){
  30. uceClient.returnResource(jedis);
  31. }
  32. }
  33. String problemCode = String.valueOf(value);
  34. for(int i = 0;i < 7;i++){
  35. if (problemCode.length() < 7){
  36. problemCode = "0"+problemCode;
  37. }else{
  38. break;
  39. }
  40. }
  41. return codePre + nowDateStr + problemCode;
  42. }

2、宕机时,调用的核心接口方法

  1. /**
  2. * 重置编码缓存
  3. * @param redisKey
  4. * @return
  5. */
  6. public static Long resetCodeRedis(String redisKey){
  7. String oldDateStr = null;
  8. String nowDateStr = DateTimeUtil.getDateTimeStr(new Date(),DateTimeUtil.DATE_PATTERN_YYYYMMDD);
  9. //编码的最大字符串
  10. String databaseMaxCode = null;
  11. //问题件
  12. if(StringUtil.isNotEmpty(redisKey) && redisKey.equals(WO_PROBLEM_CODE_KEY)){
  13. CsWoProblemService csWoProblemService = SpringContextUtil.getBean("csWoProblemService");
  14. //获取数据库中的问题件的最大编码
  15. databaseMaxCode = csWoProblemService.getMaxCode(WO_PROBLEM_CODE_PRE);
  16. //获取编码的日期部分
  17. if(StringUtil.isNotEmpty(databaseMaxCode)){
  18. oldDateStr = databaseMaxCode.substring(2,10);
  19. databaseMaxCode = databaseMaxCode.substring(10);
  20. }
  21. }else if(StringUtil.isNotEmpty(redisKey) && redisKey.equals(WO_CUST_SER_CODE_KEY)){
  22. CsWoCustSerService csWoCustSerService = SpringContextUtil.getBean("csWoCustSerService");
  23. //获取数据库中的客户服务类工单的最大编码
  24. databaseMaxCode = csWoCustSerService.getMaxCode("");
  25. //获取编码的日期部分
  26. if(StringUtil.isNotEmpty(databaseMaxCode)){
  27. oldDateStr = databaseMaxCode.substring(0,8);
  28. databaseMaxCode = databaseMaxCode.substring(8);
  29. }
  30. }
  31. Long value = getRedisValue(oldDateStr,nowDateStr,databaseMaxCode);
  32. RedisClient uceClient = null;
  33. Jedis jedisCluster = null;
  34. try {
  35. uceClient = SpringContextUtil.getBean("uceClient");
  36. jedisCluster = uceClient.getResource();
  37. boolean keyExist = jedisCluster.exists(redisKey + nowDateStr);
  38. // NX是不存在时才set, XX是存在时才set, EX是秒,PX是毫秒
  39. if (keyExist) {
  40. jedisCluster.del(redisKey + nowDateStr);
  41. }
  42. //设置缓存值,并设置为24小时后自动失效
  43. jedisCluster.set(redisKey + nowDateStr, String.valueOf((value + 1)), "NX","EX", (24*60*60+1000));
  44. }catch (Exception e){
  45. logger.error((redisKey + "编码重置异常:"),e);
  46. }finally {
  47. if(uceClient != null){
  48. uceClient.returnResource(jedisCluster);
  49. }
  50. }
  51. return value + 1;
  52. }
  53. /**
  54. * 解析redis的值
  55. * @param oldDateStr
  56. * @param nowDateStr
  57. * @param databaseMaxCode
  58. * @return
  59. */
  60. public static Long getRedisValue(String oldDateStr,String nowDateStr,String databaseMaxCode){
  61. Long value = 0L;
  62. String firstCodeZero = "0";
  63. //如果日期相同,解析编码数据存到缓存中
  64. if(StringUtil.isNotEmpty(oldDateStr) && StringUtil.isNotEmpty(nowDateStr) && oldDateStr.equals(nowDateStr) && StringUtil.isNotEmpty(databaseMaxCode)){
  65. for(int i = 0;i < 7;i++){
  66. String firstCode = databaseMaxCode.substring(0,1);
  67. if (StringUtil.isNotEmpty(firstCode) && firstCodeZero.equals(firstCode)){
  68. databaseMaxCode = databaseMaxCode.substring(1);
  69. }else{
  70. break;
  71. }
  72. }
  73. if (StringUtil.isNotEmpty(databaseMaxCode)){
  74. value = Long.parseLong(databaseMaxCode);
  75. }
  76. }
  77. return value;
  78. }

注意:jedis使用后一定要close,否则jedis连接被占用的会越来越多,可用的连接数会越来越少,最终会导致redis宕机,最终项目宕机。本项目是在finally中调用的自己封装的returnResource()方法,此方法中会进行关闭操作

发表评论

表情:
评论列表 (有 0 条评论,701人围观)

还没有评论,来说两句吧...

相关阅读

    相关 Redis命令:INCR key加1

    > 起始版本:1.0.0 > 时间复杂度:O(1) 对存储在指定key的数值执行原子的加1操作。 如果指定的key不存在,那么在执行incr操作之前,会先将它的值设定为

    相关 使用redis解决并发操作问题

    在日常的开发中,有时我们会遇到这样的场景:多个人对同一个数据进行修改操作,导致并发问题发生。这个问题可以通过悲观锁来解决,但是悲观锁也是有限制的,在某些场景中是不适应的,因为和