{"id":1246,"date":"2025-09-18T20:28:21","date_gmt":"2025-09-18T12:28:21","guid":{"rendered":"https:\/\/eve2333.top\/?p=1246"},"modified":"2025-10-02T16:38:23","modified_gmt":"2025-10-02T08:38:23","slug":"%e7%89%9b%e5%88%b8onecoupon%e7%b3%bb%e7%bb%9f-%e7%ac%ac%e2%91%a2%e7%ab%a0%e8%8a%82%ef%bc%9a%e5%88%86%e5%8f%91%e5%bc%95%e6%93%8e%e6%9c%8d%e5%8a%a1-%e4%b8%8a","status":"publish","type":"post","link":"https:\/\/eve2333.top\/?p=1246","title":{"rendered":"\u725b\u5238oneCoupon\u7cfb\u7edf \u7b2c\u2462\u7ae0\u8282\uff1a\u5206\u53d1&amp;\u5f15\u64ce\u670d\u52a1-\u4e0a"},"content":{"rendered":"\n<h2 class=\"wp-block-heading\">&nbsp;\u7b2c16\u5c0f\u8282\uff1a\u7528\u6237\u67e5\u8be2\u4f18\u60e0\u5238\u4e4b\u7f13\u5b58\u51fb\u7a7f<\/h2>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"git\">Git \u5206\u652f<\/h2>\n\n\n\n<p>20240826_dev_coupon-template-query_cache_ding.ma<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4e1a\u52a1\u80cc\u666f<\/h2>\n\n\n\n<p>\u5728\u5206\u53d1\u670d\u52a1\u4e2d\uff0c\u6211\u4eec\u9700\u8981\u8c03\u7528\u4f18\u60e0\u5238\u6a21\u677f\u7684\u76f8\u5173\u4fe1\u606f\u3002\u4e3a\u4e86\u907f\u514d\u5404\u4e2a\u670d\u52a1\u4e2d\u91cd\u590d\u5b9e\u73b0\u6a21\u677f\u67e5\u8be2\u529f\u80fd\uff0c\u6211\u4eec\u9996\u5148\u7f16\u5199\u4e86\u4e00\u4e2a\u901a\u7528\u7684\u5f15\u64ce\u5c42\u6a21\u677f\u67e5\u8be2\u65b9\u6cd5\uff0c\u4ee5\u652f\u6301 C \u7aef\u7528\u6237\u548c\u5185\u90e8\u5e94\u7528\u7684\u67e5\u8be2\u3002\u50cf\u8fd9\u79cd\u5927\u6d41\u91cf\u7684\u63a5\u53e3\uff0c\u80af\u5b9a\u662f\u9700\u8981\u653e\u5230\u7f13\u5b58\u7684\u3002<\/p>\n\n\n\n<p>\u7528\u6237\u5e38\u89c4\u8bbf\u95ee\u4f18\u60e0\u5238\u6a21\u677f\u65f6\u5e8f\u56fe\u5982\u4e0b\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"435\" width=\"326\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/fbc44bcd27cca5f489a64680eb3fe72b.png\" alt=\"\"><\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4ec0\u4e48\u662f\u7f13\u5b58\u51fb\u7a7f\uff1f<\/h2>\n\n\n\n<p>\u7f13\u5b58\u51fb\u7a7f\u6307\u5728\u9ad8\u5e76\u53d1\u7684\u7cfb\u7edf\u4e2d\uff0c<strong>\u4e00\u4e2a\u70ed\u70b9\u6570\u636e\u7f13\u5b58\u8fc7\u671f\u6216\u8005\u5728\u7f13\u5b58\u4e2d\u4e0d\u5b58\u5728<\/strong>\uff0c\u5bfc\u81f4\u5927\u91cf\u5e76\u53d1\u8bf7\u6c42\u76f4\u63a5\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u4ece\u800c\u7ed9\u6570\u636e\u5e93\u9020\u6210\u5de8\u5927\u538b\u529b\uff0c\u751a\u81f3\u53ef\u80fd\u5f15\u8d77\u5b95\u673a\u3002<\/p>\n\n\n\n<p>\u5177\u4f53\u6765\u8bf4\uff0c\u5f53\u67d0\u4e2a\u70ed\u70b9\u6570\u636e\u5728\u7f13\u5b58\u4e2d\u8fc7\u671f\u65f6\uff0c<strong>\u5982\u679c\u6b64\u65f6\u6709\u5927\u91cf\u5e76\u53d1\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u8fd9\u4e2a\u6570\u636e<\/strong>\uff0c\u7531\u4e8e\u7f13\u5b58\u4e2d\u4e0d\u5b58\u5728\uff0c\u6240\u6709\u8bf7\u6c42\u90fd\u4f1a\u76f4\u63a5\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u5bfc\u81f4\u6570\u636e\u5e93\u8d1f\u8f7d\u6025\u5267\u589e\u52a0\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"522\" width=\"531\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/89f3970c0522a1bf42b474d803197096.png\" alt=\"\"><\/p>\n\n\n\n<p>\u4f2a\u4ee3\u7801\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<p><code>public String selectTrain(String id) { String cacheData = cache.get(id); &nbsp; &nbsp;\/\/ \u67e5\u8be2\u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u53bb\u6570\u636e\u5e93\u67e5\u8be2\u5e76\u653e\u5165\u5230\u7f13\u5b58 &nbsp;if (StrUtil.isBlank(cacheData)) { &nbsp; &nbsp; &nbsp; &nbsp;\/\/ \u83b7\u53d6\u6570\u636e\u5e93\u4e2d\u5b58\u5728\u7684\u6570\u636e &nbsp; &nbsp; &nbsp; &nbsp;String dbData = trainMapper.selectId(id); &nbsp; &nbsp; &nbsp; &nbsp;if (StrUtil.isNotBlank(dbData)) { &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;\/\/ \u5c06\u67e5\u8be2\u5230\u7684\u6570\u636e\u653e\u5165\u7f13\u5b58\uff0c\u4e0b\u6b21\u67e5\u8be2\u5c31\u6709\u6570\u636e\u4e86 &nbsp; &nbsp; &nbsp;cahce.set(id, dbData); &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;cacheData = dbData; &nbsp; &nbsp; &nbsp; } &nbsp; } return cacheData; }<\/code><\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u7f13\u5b58\u51fb\u7a7f\u89e3\u51b3\u65b9\u6848<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u9884\u70ed\u548c\u7f13\u5b58\u6c38\u4e0d\u8fc7\u671f<\/h3>\n\n\n\n<p>\u4e00\u822c\u6765\u8bf4\uff0c\u6211\u4eec\u4f1a\u901a\u8fc7\u9884\u70ed\u548c\u7f13\u5b58\u6c38\u4e0d\u8fc7\u671f\u7684\u673a\u5236\u8ba9\u7f13\u5b58\u4e0d\u51fb\u7a7f\uff0c\u8fd9\u6837\u5373\u4f7f\u518d\u5927\u7684\u6d41\u91cf\u4e5f\u53ef\u4ee5\u901a\u8fc7\u7f13\u5b58\u53bb\u6297\u3002<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7f13\u5b58\u9884\u70ed\uff1a\u70ed\u70b9\u6570\u636e\u9884\u52a0\u8f7d\uff0c\u6307\u7684\u662f\u5728\u6d3b\u52a8\u6216\u8005\u5927\u4fc3\u5f00\u59cb\u524d\uff0c<strong>\u9488\u5bf9\u5df2\u77e5\u7684\u70ed\u70b9\u6570\u636e\u4ece\u6570\u636e\u5e93\u52a0\u8f7d\u5230\u7f13\u5b58\u4e2d<\/strong>\uff0c\u8fd9\u6837\u53ef\u4ee5\u907f\u514d\u6d77\u91cf\u8bf7\u6c42\u7b2c\u4e00\u6b21\u8bbf\u95ee\u70ed\u70b9\u6570\u636e\u9700\u8981\u4ece\u6570\u636e\u5e93\u8bfb\u53d6\u7684\u6d41\u7a0b\u3002<\/li>\n\n\n\n<li>\u6c38\u4e0d\u8fc7\u671f\uff1a\u70ed\u70b9\u6570\u636e\u6c38\u4e0d\u8fc7\u671f\uff0c\u6307\u7684\u5c31\u662f\u53ef\u4ee5\u9884\u77e5\u7684\u70ed\u70b9\u6570\u636e\uff0c<strong>\u5728\u6d3b\u52a8\u5f00\u59cb\u524d\uff0c\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\u4e3a -1<\/strong>\u3002\u8fd9\u6837\u7684\u8bdd\uff0c\u5c31\u4e0d\u4f1a\u6709\u7f13\u5b58\u51fb\u7a7f\u7684\u98ce\u9669\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u4e0a\u9762\u4e24\u4e2a\u4e00\u822c\u90fd\u662f\u642d\u914d\u4e00\u8d77\u4f7f\u7528\u7684\u3002\u7b49\u5bf9\u5e94\u70ed\u70b9\u7f13\u5b58\u7684\u6d3b\u52a8\u7ed3\u675f\u540e\uff0c\u8fd9\u4e9b\u6570\u636e\u8bbf\u95ee\u91cf\u5c31\u6bd4\u8f83\u4f4e\u4e86\uff0c\u53ef\u4ee5\u901a\u8fc7\u540e\u53f0\u4efb\u52a1\u7684\u65b9\u6848\u5bf9\u6307\u5b9a\u7f13\u5b58\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\uff0c\u8fd9\u6837\u53ef\u4ee5\u6709\u6548\u964d\u4f4e Redis \u5b58\u50a8\u538b\u529b\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"818\" width=\"457\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/2699357f6a49c8cede8f4a6f5d46dee1.png\" alt=\"\"><\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u5206\u5e03\u5f0f\u9501\u4e4b\u53cc\u91cd\u5224\u5b9a\u9501<\/h3>\n\n\n\n<p>\u5206\u5e03\u5f0f\u9501\u7684\u89e3\u51b3\u65b9\u6848\u5c31\u662f\u4fdd\u8bc1<strong>\u53ea\u6709\u4e00\u4e2a\u8bf7\u6c42\u53ef\u4ee5\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u5176\u5b83\u8bf7\u6c42\u7b49\u5f85\u7ed3\u679c<\/strong>\u3002\u8fd9\u6837\u53ef\u4ee5\u907f\u514d\u5927\u91cf\u7684\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u6570\u636e\u5e93\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"561\" width=\"464\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/f447a2ad18495763f92decc75f39d9cd.png\" alt=\"\"><\/p>\n\n\n\n<p>\u5728\u539f\u6709\u57fa\u7840\u4e0a\u7ee7\u7eed\u6539\u8fdb\uff0c\u4f2a\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectTrain(String id) {\n    String cacheData = cache.get(id); \/\/ \u67e5\u8be2\u7f13\u5b58\n    if (StrUtil.isBlank(cacheData)) { \/\/ \u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u53bb\u6570\u636e\u5e93\u67e5\u8be2\u5e76\u653e\u5165\u7f13\u5b58\n        Lock lock = getLock(id); \/\/ \u83b7\u53d6\u5206\u5e03\u5f0f\u9501\n        lock.lock();\n        try {\n            String dbData = trainMapper.selectId(id); \/\/ \u67e5\u8be2\u6570\u636e\u5e93\n            if (StrUtil.isNotBlank(dbData)) { \/\/ \u6570\u636e\u5e93\u4e2d\u6709\u6570\u636e\n                cache.set(id, dbData); \/\/ \u5c06\u6570\u636e\u653e\u5165\u7f13\u5b58\n                cacheData = dbData;\n            }\n        } finally {\n            lock.unlock(); \/\/ \u91ca\u653e\u9501\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u4f46\u662f\u8fd9\u79cd\u7684\u8bdd\u6709\u4e00\u4e2a\u5f0a\u7aef\uff0c\u90a3\u5c31\u662f<strong>\u83b7\u53d6\u5206\u5e03\u5f0f\u9501\u7684\u8bf7\u6c42\uff0c\u90fd\u4f1a\u6267\u884c\u4e00\u904d\u67e5\u8be2\u6570\u636e\u5e93<\/strong>\uff0c\u5e76\u66f4\u65b0\u5230\u7f13\u5b58\u3002\u7406\u8bba\u4e0a\u53ea\u6709<strong>\u7b2c\u4e00\u4e2a\u52a0\u8f7d\u6570\u636e\u5e93\u8bb0\u5f55\u8bf7\u6c42\u662f\u6709\u6548\u7684<\/strong>\u3002<\/p>\n\n\n\n<p>\u9488\u5bf9\u8fd9\u4e2a\u95ee\u9898\uff0c\u53ef\u4ee5\u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u7684\u5f62\u5f0f\uff0c<strong>\u5728\u83b7\u53d6\u5230\u5206\u5e03\u5f0f\u9501\u4e4b\u540e\uff0c\u518d\u6b21\u67e5\u8be2\u4e00\u6b21\u7f13\u5b58\u662f\u5426\u5b58\u5728<\/strong>\u3002\u5982\u679c\u7f13\u5b58\u4e2d\u5b58\u5728\u6570\u636e\uff0c\u5c31\u76f4\u63a5\u8fd4\u56de\uff1b\u5982\u679c\u4e0d\u5b58\u5728\uff0c\u624d\u7ee7\u7eed\u6267\u884c\u67e5\u8be2\u6570\u636e\u5e93\u7684\u64cd\u4f5c\u3002\u8fd9\u6837\u5c31\u53ef\u4ee5\u907f\u514d\u5927\u91cf\u8bf7\u6c42\u8bbf\u95ee\u6570\u636e\u5e93\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"754\" width=\"484\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/76faba6e64e865ead6c6b475291dd22c.png\" alt=\"\"><\/p>\n\n\n\n<p>\u4f2a\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectTrain(String id) {\n    \/\/ \u67e5\u8be2\u7f13\u5b58\n    String cacheData = cache.get(id);\n    \/\/ \u67e5\u8be2\u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u53bb\u6570\u636e\u5e93\u67e5\u8be2\u5e76\u653e\u5165\u5230\u7f13\u5b58\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u4e3a\u907f\u514d\u5927\u91cf\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u901a\u8fc7\u5206\u5e03\u5f0f\u9501\u51cf\u5c11\u6570\u636e\u5e93\u8bbf\u95ee\u91cf\n        Lock lock = getLock(id);\n        lock.lock();\n        try {\n            \/\/ \u83b7\u53d6\u9501\u540e\u53cc\u91cd\u5224\u5b9a\n            cacheData = cache.get(id);\n            \/\/ \u7406\u8bba\u4e0a\u53ea\u6709\u7b2c\u4e00\u4e2a\u8bf7\u6c42\u52a0\u8f7d\u6570\u636e\u5e93\u662f\u6709\u6548\u7684\uff0c\u56e0\u4e3a\u5b83\u52a0\u8f7d\u540e\u4f1a\u628a\u6570\u636e\u653e\u5230\u7f13\u5b58\n            \/\/ \u540e\u9762\u7684\u8bf7\u6c42\u518d\u8bf7\u6c42\u6570\u636e\u5e93\u52a0\u8f7d\u7f13\u5b58\u5c31\u6ca1\u6709\u5fc5\u8981\u4e86\n            if (StrUtil.isBlank(cacheData)) {\n                \/\/ \u83b7\u53d6\u6570\u636e\u5e93\u4e2d\u5b58\u5728\u7684\u6570\u636e\n                String dbData = trainMapper.selectId(id);\n                if (StrUtil.isNotBlank(dbData)) {\n                    \/\/ \u5c06\u67e5\u8be2\u5230\u7684\u6570\u636e\u653e\u5165\u7f13\u5b58\uff0c\u4e0b\u6b21\u67e5\u8be2\u5c31\u6709\u6570\u636e\u4e86\n                    cache.set(id, dbData);\n                    cacheData = dbData;\n                }\n            }\n        } finally {\n            lock.unlock();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u4e0b\u9762\u662f\u8fd9\u79cd\u573a\u666f\u4e0b\u89e3\u51b3\u65b9\u6848\u7684\u4e00\u822c\u6b65\u9aa4\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u83b7\u53d6\u9501\uff1a\u5728\u67e5\u8be2\u6570\u636e\u5e93\u524d\uff0c\u9996\u5148\u5c1d\u8bd5\u83b7\u53d6\u4e00\u4e2a\u5206\u5e03\u5f0f\u9501\u3002\u53ea\u6709\u4e00\u4e2a\u7ebf\u7a0b\u80fd\u591f\u6210\u529f\u83b7\u53d6\u9501\uff0c\u5176\u4ed6\u7ebf\u7a0b\u9700\u8981\u7b49\u5f85\uff1b<\/li>\n\n\n\n<li>2. \u67e5\u8be2\u6570\u636e\u5e93\uff1a\u5982\u679c\u53cc\u91cd\u5224\u65ad\u786e\u8ba4\u6570\u636e\u786e\u5b9e\u4e0d\u5b58\u5728\u4e8e\u7f13\u5b58\u4e2d\uff0c\u90a3\u4e48\u5c31\u6267\u884c\u67e5\u8be2\u6570\u636e\u5e93\u7684\u64cd\u4f5c\uff0c\u83b7\u53d6\u6570\u636e\uff1b<\/li>\n\n\n\n<li>3. \u5199\u5165\u7f13\u5b58\uff1a\u83b7\u53d6\u5230\u6570\u636e\u540e\uff0c\u5c06\u6570\u636e\u5199\u5165\u7f13\u5b58\uff0c\u5e76\u8bbe\u7f6e\u4e00\u4e2a\u5408\u9002\u7684\u8fc7\u671f\u65f6\u95f4\uff0c\u4ee5\u9632\u6b62\u7f13\u5b58\u6c38\u8fdc\u4e0d\u4f1a\u88ab\u66f4\u65b0\uff1b<\/li>\n\n\n\n<li>4. \u91ca\u653e\u9501\uff1a\u6700\u540e\uff0c\u91ca\u653e\u83b7\u53d6\u7684\u9501\uff0c\u4ee5\u4fbf\u5176\u4ed6\u7ebf\u7a0b\u53ef\u4ee5\u7ee7\u7eed\u4f7f\u7528\u8fd9\u4e2a\u9501\u3002<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u9ad8\u5e76\u53d1\u6781\u7aef\u60c5\u51b5<\/h3>\n\n\n\n<p>\u5f88\u591a\u540c\u5b66\u8ba4\u4e3a\u5230\u8fd9\u91cc\u5c31\u7ed3\u675f\u4e86\uff0c\u4f46\u8fd9\u6070\u6070\u53ea\u662f\u5f00\u59cb\uff0c\u771f\u6b63\u96be\u5f97\u662f\u63a5\u4e0b\u6765\u8981\u8bb2\u7684\u3002<\/p>\n\n\n\n<p>\u6211\u4e3e\u4e2a\u573a\u666f\uff0c<strong>\u6709\u4e00\u4e07\u4e2a\u8bf7\u6c42\u540c\u4e00\u65f6\u95f4\u8bbf\u95ee\u89e6\u53d1\u4e86\u7f13\u5b58\u51fb\u7a7f<\/strong>\uff0c\u5982\u679c\u7528\u53cc\u91cd\u5224\u5b9a\u9501\uff0c\u903b\u8f91\u662f\u8fd9\u6837\u7684\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u7b2c\u4e00\u4e2a\u8bf7\u6c42\u52a0\u9501\u3001\u67e5\u8be2\u7f13\u5b58\u662f\u5426\u5b58\u5728\u3001\u67e5\u8be2\u6570\u636e\u5e93\u3001\u653e\u5165\u7f13\u5b58\u3001\u89e3\u9501\uff0c\u5047\u8bbe\u6211\u4eec\u7528\u4e8650\u6beb\u79d2\uff1b<\/li>\n\n\n\n<li>2. \u7b2c\u4e8c\u4e2a\u8bf7\u6c42\u62ff\u5230\u9501\u67e5\u8be2\u7f13\u5b58\u3001\u89e3\u9501\u7528\u4e861\u6beb\u79d2\uff1b<\/li>\n\n\n\n<li>3. \u90a3\u6700\u540e\u4e00\u4e2a\u8bf7\u6c42\u9700\u8981\u7b49\u5f8510049\u6beb\u79d2\u540e\u624d\u80fd\u8fd4\u56de\uff0c\u7528\u6237\u7b49\u5f85\u65f6\u95f4\u8fc7\u957f\uff0c\u6781\u7aef\u60c5\u51b5\u4e0b\u53ef\u80fd\u4f1a\u89e6\u53d1\u5e94\u7528\u7684\u5185\u5b58\u6ea2\u51fa\u3002<\/li>\n<\/ul>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-1-try-lock\">3.1 \u5c1d\u8bd5\u83b7\u53d6\u9501 tryLock<\/h4>\n\n\n\n<p>\u50cf\u4e0a\u9762\u8fd9\u79cd\u573a\u666f\uff0c\u7c7b\u4f3c\u4e8e\u79d2\u6740\u7684\u67b6\u6784\uff0c\u6211\u4eec\u8981\u505a\u7684\u5c31\u662f\u4e0d\u8ba9\u7528\u6237\u8bf7\u6c42\u5728\u670d\u52a1\u7aef\u963b\u585e\u8fc7\u957f\u65f6\u95f4\u3002\u90a3\u5c31\u53ef\u4ee5\u4f7f\u7528\u5c1d\u8bd5\u83b7\u53d6\u9501 <code>tryLock<\/code> API\uff0c\u5b83\u7684\u8bed\u4e49\u662f\u5982\u679c\u62ff\u9501\u5931\u8d25\u76f4\u63a5\u8fd4\u56de\uff0c\u800c\u4e0d\u662f\u963b\u585e\u7b49\u5f85\u76f4\u5230\u83b7\u53d6\u9501\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectTrain(String id) {\n    \/\/ \u67e5\u8be2\u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u53bb\u6570\u636e\u5e93\u67e5\u8be2\u5e76\u653e\u5165\u5230\u7f13\u5b58\n    String cacheData = cache.get(id);\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u4e3a\u907f\u514d\u5927\u91cf\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u901a\u8fc7\u5206\u5e03\u5f0f\u9501\u51cf\u5c11\u6570\u636e\u5e93\u8bbf\u95ee\u91cf\n        Lock lock = getLock(id);\n        \/\/ \u5c1d\u8bd5\u83b7\u53d6\u9501\uff0c\u83b7\u53d6\u5931\u8d25\u76f4\u63a5\u8fd4\u56de\u7528\u6237\u8bf7\u6c42\uff0c\u5e76\u63d0\u9192\u7528\u6237\u7a0d\u540e\u518d\u8bd5\n        if (!lock.tryLock()) {\n            throw new RuntimeException(\"\u5f53\u524d\u8bbf\u95ee\u4eba\u6570\u8fc7\u591a\uff0c\u8bf7\u7a0d\u5019\u518d\u8bd5...\");\n        }\n        try {\n            \/\/ \u83b7\u53d6\u6570\u636e\u5e93\u4e2d\u5b58\u5728\u7684\u6570\u636e\n            String dbData = trainMapper.selectId(id);\n            if (StrUtil.isNotBlank(dbData)) {\n                \/\/ \u5c06\u67e5\u8be2\u5230\u7684\u6570\u636e\u653e\u5165\u7f13\u5b58\uff0c\u4e0b\u6b21\u67e5\u8be2\u5c31\u6709\u6570\u636e\u4e86\n                cache.set(id, dbData);\n                cacheData = dbData;\n            }\n        } finally {\n            lock.unlock();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u901a\u8fc7\u8fd9\u79cd\u65b9\u5f0f\u6211\u4eec\u53ef\u4ee5\u5feb\u901f\u5931\u8d25\uff0c\u544a\u8bc9\u7528\u6237\u7f51\u7edc\u5f02\u5e38\u8bf7\u7a0d\u540e\u518d\u8bd5\uff0c\u7b49\u7528\u6237\u518d\u5c1d\u8bd5\u5237\u65b0\u7684\u65f6\u5019\uff0c\u5176\u5b9e\u83b7\u53d6\u9501\u7684\u7ebf\u7a0b\u5df2\u7ecf\u628a\u6570\u636e\u653e\u5230\u4e86\u7f13\u5b58\u3002<\/p>\n\n\n\n<p>\u56e0\u4e3a\u8fd9\u79cd\u65b9\u6848\u5bf9\u7528\u6237\u64cd\u4f5c\u4f53\u9a8c\u4e0d\u53cb\u597d\uff0c\u6240\u4ee5\u4e5f\u53ea\u662f\u9002\u7528\u4e8e\u90e8\u5206\u573a\u666f\u3002\u5728\u5b9e\u9645\u5f00\u53d1\u4e2d\uff0c\u9700\u8981\u7075\u6d3b\u53d8\u66f4\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-2\">3.2 \u5206\u5e03\u5f0f\u9501\u5206\u7247<\/h4>\n\n\n\n<p>\u8fd8\u6709\u4e00\u79cd\u6bd4\u8f83\u4f18\u96c5\u7684\u89e3\u51b3\u65b9\u6848\u662f\u901a\u8fc7\u5206\u5e03\u5f0f\u9501\u5206\u7247\u7684\u5f62\u5f0f\uff0c\u8ba9\u5e76\u884c\u7684\u7ebf\u7a0b\u66f4\u591a\u4e00\u4e9b\u3002\u56e0\u4e3a\u540c\u4e00\u65f6\u95f4\u6709\u591a\u4e2a\u7ebf\u7a0b\u80fd\u540c\u65f6\u64cd\u4f5c\uff0c\u6240\u4ee5\u7406\u8bba\u4e0a\uff0c\u8bbe\u7f6e\u5206\u7247\u91cf\u7684\u591a\u5c11\uff0c\u4e5f\u5c31\u662f\u6027\u80fd\u63d0\u5347\u4e86\u8fd1\u591a\u5c11\u500d\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"1076\" width=\"759\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/32996d60607760a4089e5b610afb3d59.png\" alt=\"\"><\/p>\n\n\n\n<p>\u4f2a\u4ee3\u7801\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectTrain(String id, String userId) {\n    \/\/ \u67e5\u8be2\u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u53bb\u6570\u636e\u5e93\u67e5\u8be2\u5e76\u653e\u5165\u5230\u7f13\u5b58\n    String cacheData = cache.get(id);\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u5047\u8bbe\u8bbe\u7f6e10\u628a\u5206\u5e03\u5f0f\u9501\uff0c\u90a3\u4e48\u5c31\u901a\u8fc7\u552f\u4e00\u6807\u8bc6\uff08\u8fd9\u91cc\u53d6\u7528\u6237ID\uff09\u8fdb\u884c\u53d6\u6a21\u83b7\u53d6\u5206\u7247\u4e0b\u6807\n        int idx = Math.abs(userId.hashCode()) % 10;\n        \/\/ \u4e3a\u907f\u514d\u5927\u91cf\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u901a\u8fc7\u5206\u5e03\u5f0f\u9501\u51cf\u5c11\u6570\u636e\u5e93\u8bbf\u95ee\u91cf\n        Lock lock = getLock(id + idx);\n        lock.lock();\n        try {\n            \/\/ \u83b7\u53d6\u9501\u540e\u53cc\u91cd\u5224\u5b9a\n            cacheData = cache.get(id);\n            \/\/ \u7406\u8bba\u4e0a\u53ea\u6709\u7b2c\u4e00\u4e2a\u8bf7\u6c42\u52a0\u8f7d\u6570\u636e\u5e93\u662f\u6709\u6548\u7684\uff0c\u56e0\u4e3a\u5b83\u52a0\u8f7d\u540e\u4f1a\u628a\u6570\u636e\u653e\u5230\u7f13\u5b58\n            \/\/ \u540e\u9762\u7684\u8bf7\u6c42\u518d\u8bf7\u6c42\u6570\u636e\u5e93\u52a0\u8f7d\u7f13\u5b58\u5c31\u6ca1\u6709\u5fc5\u8981\u4e86\n            if (StrUtil.isBlank(cacheData)) {\n                \/\/ \u83b7\u53d6\u6570\u636e\u5e93\u4e2d\u5b58\u5728\u7684\u6570\u636e\n                String dbData = trainMapper.selectId(id);\n                if (StrUtil.isNotBlank(dbData)) {\n                    \/\/ \u5c06\u67e5\u8be2\u5230\u7684\u6570\u636e\u653e\u5165\u7f13\u5b58\uff0c\u4e0b\u6b21\u67e5\u8be2\u5c31\u6709\u6570\u636e\u4e86\n                    cache.set(id, dbData);\n                    cacheData = dbData;\n                }\n            }\n        } finally {\n            lock.unlock();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<h2 class=\"wp-block-heading\">\u5f00\u53d1\u4f18\u60e0\u5238\u6a21\u677f\u67e5\u8be2<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u5e38\u89c4\u4f18\u60e0\u5238\u6a21\u677f\u67e5\u8be2<\/h3>\n\n\n\n<p>\u7b2c\u4e00\u7248\u672c\u7684\u4ee3\u7801\u662f\u6700\u4e3a\u539f\u59cb\u7684\u76f4\u63a5\u67e5\u8be2\u6570\u636e\u5e93\uff0c\u6570\u636e\u91cf\u5927\u7684\u65f6\u5019\uff0c\u80af\u5b9a\u662f\u6ca1\u529e\u6cd5\u625b\u4f4f\u7684\uff0c\u6240\u4ee5\u6211\u4eec\u63a5\u4e0b\u6765\u5f15\u5165\u89e3\u51b3\u4e86\u7f13\u5b58\u51fb\u7a7f\u7684\u7248\u672c\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>package com.nageoffer.onecoupon.engine.service.impl;\n\nimport cn.hutool.core.bean.BeanUtil;\nimport com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;\nimport com.baomidou.mybatisplus.core.toolkit.Wrappers;\nimport com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;\nimport com.nageoffer.onecoupon.engine.common.enums.CouponTemplateStatusEnum;\nimport com.nageoffer.onecoupon.engine.dao.entity.CouponTemplateDO;\nimport com.nageoffer.onecoupon.engine.dao.mapper.CouponTemplateMapper;\nimport com.nageoffer.onecoupon.engine.dto.req.CouponTemplateQueryReqDTO;\nimport com.nageoffer.onecoupon.engine.dto.resp.CouponTemplateQueryRespDTO;\nimport com.nageoffer.onecoupon.engine.service.CouponTemplateService;\nimport lombok.RequiredArgsConstructor;\nimport lombok.extern.slf4j.Slf4j;\nimport org.springframework.stereotype.Service;\n\n\/**\n * \u4f18\u60e0\u5238\u6a21\u677f\u4e1a\u52a1\u903b\u8f91\u5b9e\u73b0\u5c42\n * &lt;p&gt;\n * \u4f5c\u8005\uff1a\u9a6c\u4e01\n * \u52a0\u9879\u76ee\u7fa4\uff1a\u65e9\u52a0\u5165\u5c31\u662f\u4f18\u52bf\uff01500\u4eba\u5185\u90e8\u9879\u76ee\u7fa4\uff0c\u5206\u4eab\u7684\u77e5\u8bc6\u603b\u6709\u4f60\u9700\u8981\u7684 &lt;a href=\"https:\/\/t.zsxq.com\/cw7b9 \" \/&gt;\n * \u5f00\u53d1\u65f6\u95f4\uff1a2024-07-08\n *\/\n@Slf4j\n@Service\n@RequiredArgsConstructor\npublic class CouponTemplateServiceImpl extends ServiceImpl&lt;CouponTemplateMapper, CouponTemplateDO&gt; implements CouponTemplateService {\n\n    private final CouponTemplateMapper couponTemplateMapper;\n\n    @Override\n    public CouponTemplateQueryRespDTO findCouponTemplate(CouponTemplateQueryReqDTO requestParam) {\n        LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n        CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n        return BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n    }\n}<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u67e5\u8be2\u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58<\/h3>\n\n\n\n<p>\u6211\u4eec\u7701\u53bb\u4e86\u7b80\u5355\u7684\u5206\u5e03\u5f0f\u8fc7\u7a0b\uff0c\u4e0a\u9762\u5df2\u7ecf\u6709\u7b80\u5355\u7684\u793a\u4f8b\uff0c\u76f4\u63a5\u7528\u8f83\u5e38\u89c1\u7684\u53cc\u91cd\u5224\u5b9a\u9501\u3002<\/p>\n\n\n\n<p>@Override<br>public CouponTemplateQueryRespDTO findCouponTemplate(CouponTemplateQueryReqDTO requestParam) {<br>\/\/ \u67e5\u8be2 Redis \u7f13\u5b58\u4e2d\u662f\u5426\u5b58\u5728\u4f18\u60e0\u5238\u6a21\u677f\u4fe1\u606f<br>String couponTemplateCacheKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId());<br>Map couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u5982\u679c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u9700\u8981\u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u7684\u5f62\u5f0f\u8bfb\u53d6\u6570\u636e\u5e93\u4e2d\u7684\u8bb0\u5f55\nif (MapUtil.isEmpty(couponTemplateCacheMap)) {\n    \/\/ \u83b7\u53d6\u4f18\u60e0\u5238\u6a21\u677f\u5206\u5e03\u5f0f\u9501\n    \/\/ \u5173\u4e8e\u7f13\u5b58\u51fb\u7a7f\u66f4\u591a\u6ce8\u91ca\u4e8b\u9879\uff0c\u6b22\u8fce\u67e5\u770b\u6211\u7684B\u7ad9\u89c6\u9891\uff1ahttps:\/\/www.bilibili.com\/video\/BV1qz421z7vC\n    RLock lock = redissonClient.getLock(String.format(EngineRedisConstant.LOCK_COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId()));\n    lock.lock();\n\n    try {\n        \/\/ \u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u4f18\u5316\u5927\u91cf\u8bf7\u6c42\u65e0\u610f\u4e49\u67e5\u8be2\u6570\u636e\u5e93\n        couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n        if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n            LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                    .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                    .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                    .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n            CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\n            \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u8005\u5df2\u8fc7\u671f\u76f4\u63a5\u629b\u51fa\u5f02\u5e38\n            if (couponTemplateDO == null) {\n                throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u5df2\u8fc7\u671f\");\n            }\n\n            \/\/ \u901a\u8fc7\u5c06\u6570\u636e\u5e93\u7684\u8bb0\u5f55\u5e8f\u5217\u5316\u6210 JSON \u5b57\u7b26\u4e32\u653e\u5165 Redis \u7f13\u5b58\n            CouponTemplateQueryRespDTO actualRespDTO = BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n            Map&lt;String, Object&gt; cacheTargetMap = BeanUtil.beanToMap(actualRespDTO, false, true);\n            Map&lt;String, String&gt; actualCacheTargetMap = cacheTargetMap.entrySet().stream()\n                    .collect(Collectors.toMap(\n                            Map.Entry::getKey,\n                            entry -&gt; entry.getValue() != null ? entry.getValue().toString() : \"\"\n                    ));\n\n            \/\/ \u901a\u8fc7 LUA \u811a\u672c\u6267\u884c\u8bbe\u7f6e Hash \u6570\u636e\u4ee5\u53ca\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\n            String luaScript = \"redis.call('HMSET', KEYS&#91;1], unpack(ARGV, 1, #ARGV - 1)) \" +\n                    \"redis.call('EXPIREAT', KEYS&#91;1], ARGV&#91;#ARGV])\";\n\n            List&lt;String&gt; keys = Collections.singletonList(couponTemplateCacheKey);\n            List&lt;String&gt; args = new ArrayList&lt;&gt;(actualCacheTargetMap.size() * 2 + 1);\n            actualCacheTargetMap.forEach((key, value) -&gt; {\n                args.add(key);\n                args.add(value);\n            });\n\n            \/\/ \u4f18\u60e0\u5238\u6d3b\u52a8\u8fc7\u671f\u65f6\u95f4\u8f6c\u6362\u4e3a\u79d2\u7ea7\u522b\u7684 Unix \u65f6\u95f4\u6233\n            args.add(String.valueOf(couponTemplateDO.getValidEndTime().getTime() \/ 1000));\n\n            \/\/ \u6267\u884c LUA \u811a\u672c\n            stringRedisTemplate.execute(\n                    new DefaultRedisScript&lt;&gt;(luaScript, Long.class),\n                    keys,\n                    args.toArray()\n            );\n\n            couponTemplateCacheMap = cacheTargetMap.entrySet()\n                    .stream()\n                    .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));\n        }\n    } finally {\n        lock.unlock();\n    }\n}\n\nreturn BeanUtil.mapToBean(couponTemplateCacheMap, CouponTemplateQueryRespDTO.class, false, CopyOptions.create());<\/code><\/pre>\n\n\n\n<p>}<\/p>\n\n\n\n<p>\u770b\u4e86\u4e0a\u9762\u7f13\u5b58\u51fb\u7a7f\u7406\u8bba\u77e5\u8bc6\u540e\uff0c\u8fd9\u4e2a\u4ee3\u7801\u5c31\u6bd4\u8f83\u5bb9\u6613\u7406\u89e3\u3002\u5176\u4e2d\u56e0\u4e3a\u6211\u4eec\u8981\u8fdb\u884c\u51c6\u6362 Hash putAll \u65b9\u6cd5\u6240\u4ee5\u9700\u8981\u8fdb\u884c\u591a\u6b21 Map \u8f6c\u6362\uff0c\u8fd9\u4e2a\u5c5e\u4e8e\u662f\u4e1a\u52a1\u590d\u6742\u6027\u5bfc\u81f4\uff0c\u5982\u679c\u53ea\u662f\u7b80\u5355\u653e\u4e2a String \u4ee3\u7801\u6d41\u7a0b\u4f1a\u7b80\u5355\u5f88\u591a\u3002<\/p>\n\n\n\n<p>\u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u51fb\u7a7f\u67e5\u8be2\u65f6\u5e8f\u56fe\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"856\" width=\"563\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/b6b9c2a046f4fbf6d59d246fcc61e71a.png\" alt=\"\">\u200b<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6587\u672b\u603b\u7ed3<\/h2>\n\n\n\n<p>\u867d\u7136\u6211\u4eec\u8bb2\u4e86\u591a\u79cd\u7f13\u5b58\u51fb\u7a7f\u7684\u65b9\u6848\uff0c\u5176\u4e2d\u8bf4\u5230\u7684 tryLock \u548c\u5206\u5e03\u5f0f\u9501\u5206\u7247\u4e00\u822c\u53ea\u4f1a\u51fa\u73b0\u5728\u9762\u8bd5\u4e2d\uff0c\u56e0\u4e3a\u56fd\u5185 99.99% \u7684\u516c\u53f8\u4e0d\u5b58\u5728\u5b58\u5728\u9ad8\u5e76\u53d1\u6781\u7aef\u7684\u60c5\u51b5\uff0c\u5199\u4e2a\u53cc\u91cd\u5224\u5b9a\u9501\u5df2\u7ecf\u662f\u5bf9\u4e1a\u52a1\u7684\u5c0a\u91cd\u4e86\u3002\u53ef\u4ee5\u80cc\u516b\u80a1\uff0c\u4f46\u522b\u88ab\u516b\u80a1\u5e26\u8fdb\u53bb\u3002<\/p>\n\n\n\n<p><strong>\u4f18\u60e0\u5238\u6a21\u677f\u6709\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4\u3001\u7ed3\u675f\u65f6\u95f4\u4e24\u4e2a\u5b57\u6bb5\uff0c\u6240\u4ee5\u628a\u7f13\u5b58\u7684\u8fc7\u671f\u65f6\u95f4\u8bbe\u7f6e\u6210\u548c\u4f18\u60e0\u5238\u6a21\u677f\u6709\u6548\u671f\u7684\u7ed3\u675f\u65f6\u95f4\u4e00\u81f4\uff0c\u67e5\u4e0d\u5230\u7f13\u5b58\u5c31\u76f4\u63a5\u63d0\u793a\u7528\u6237\u4f18\u60e0\u5238\u6a21\u677f\u5df2\u7ecf\u5931\u6548\uff0c\u662f\u4e0d\u662f\u5c31\u80fd\u4e0d\u7528\u8003\u8651\u7f13\u5b58\u51fb\u7a7f\u95ee\u9898\u4e86<\/strong><\/p>\n\n\n\n<p><strong>\u6211\u60f3\u53ef\u80fd\u6709\u4e9b\u573a\u666f\u4e0b\u67d0\u4e2a\u4f18\u60e0\u5238\u53ea\u80fd\u5728\u6709\u6548\u671f\u65f6\u95f4\u5185\u7684\u6307\u5b9a\u65f6\u95f4\u4f7f\u7528\uff0c\u6240\u4ee5\u8003\u8651\u5230\u8fd9\u70b9\u5c31\u6ca1\u7528\u4f60\u8bf4\u7684\u65b9\u6cd5\uff1f<\/strong><\/p>\n\n\n\n<p><strong>\u5982\u679cRedis\u7684\u5185\u5b58\u8d85\u51fa\u6700\u5927\u5185\u5b58\uff0c\u4ecd\u6709\u6548\u4f18\u60e0\u5238\u6a21\u677f \u5b58\u5728\u88ab\u6dd8\u6c70\u7684\u98ce\u9669<\/strong><\/p>\n\n\n\n<p><strong>\u7f13\u5b58\u4e2d\u7684\u4f18\u60e0\u5238\u6a21\u677f\u8bb0\u5f55\u8fc7\u671f\u65f6\u95f4\u5728\u521b\u5efa\u7684\u65f6\u5019\u5c31\u5df2\u7ecf\u8bbe\u7f6e\u4e86 \u4e3a\u4e86\u9884\u9632redis\u7279\u5b9a\u70ed\u70b9key\u5931\u6548(\u4e0d\u4e00\u5b9a\u662f\u5230\u8fbe\u8fc7\u671f\u65f6\u95f4) \u5bfc\u81f4\u7f13\u5b58\u51fb\u7a7f\u95ee\u9898 \u6240\u4ee5\u624d\u9700\u8981\u8003\u8651<\/strong><\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\n@Slf4j\n@Service\n@RequiredArgsConstructor\npublic class CouponTemplateServiceImpl extends ServiceImpl&lt;CouponTemplateMapper, CouponTemplateDO&gt; implements CouponTemplateService {\n\n    private final CouponTemplateMapper couponTemplateMapper;\n    private final StringRedisTemplate stringRedisTemplate;\n    private final RedissonClient redissonClient;\n\n    @Override\n    public CouponTemplateQueryRespDTO findCouponTemplate(CouponTemplateQueryReqDTO requestParam) {\n        \/\/ \u67e5\u8be2 Redis \u7f13\u5b58\u4e2d\u662f\u5426\u5b58\u5728\u4f18\u60e0\u5238\u6a21\u677f\u4fe1\u606f\n        String couponTemplateCacheKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId());\n        Map&lt;Object, Object&gt; couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n\n        \/\/ \u5982\u679c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u9700\u8981\u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u7684\u5f62\u5f0f\u8bfb\u53d6\u6570\u636e\u5e93\u4e2d\u7684\u8bb0\u5f55\n        if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n            \/\/ \u83b7\u53d6\u4f18\u60e0\u5238\u6a21\u677f\u5206\u5e03\u5f0f\u9501\n            \/\/ \u5173\u4e8e\u7f13\u5b58\u51fb\u7a7f\u66f4\u591a\u6ce8\u91ca\u4e8b\u9879\uff0c\u6b22\u8fce\u67e5\u770b\u6211\u7684B\u7ad9\u89c6\u9891\uff1ahttps:\/\/www.bilibili.com\/video\/BV1qz421z7vC\n            RLock lock = redissonClient.getLock(String.format(EngineRedisConstant.LOCK_COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId()));\n            lock.lock();\n\n            try {\n                \/\/ \u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u4f18\u5316\u5927\u91cf\u8bf7\u6c42\u65e0\u610f\u4e49\u67e5\u8be2\u6570\u636e\u5e93\n                couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n                if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n                    LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                            .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                            .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                            .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n                    CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\n                    \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u8005\u5df2\u8fc7\u671f\u76f4\u63a5\u629b\u51fa\u5f02\u5e38\n                    if (couponTemplateDO == null) {\n                        throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u5df2\u8fc7\u671f\");\n                    }\n\n                    \/\/ \u901a\u8fc7\u5c06\u6570\u636e\u5e93\u7684\u8bb0\u5f55\u5e8f\u5217\u5316\u6210 JSON \u5b57\u7b26\u4e32\u653e\u5165 Redis \u7f13\u5b58\n                    CouponTemplateQueryRespDTO actualRespDTO = BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n                    Map&lt;String, Object&gt; cacheTargetMap = BeanUtil.beanToMap(actualRespDTO, false, true);\n                    Map&lt;String, String&gt; actualCacheTargetMap = cacheTargetMap.entrySet().stream()\n                            .collect(Collectors.toMap(\n                                    Map.Entry::getKey,\n                                    entry -&gt; entry.getValue() != null ? entry.getValue().toString() : \"\"\n                            ));\n\n                    \/\/ \u901a\u8fc7 LUA \u811a\u672c\u6267\u884c\u8bbe\u7f6e Hash \u6570\u636e\u4ee5\u53ca\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\n                    String luaScript = \"redis.call('HMSET', KEYS&#91;1], unpack(ARGV, 1, #ARGV - 1)) \" +\n                            \"redis.call('EXPIREAT', KEYS&#91;1], ARGV&#91;#ARGV])\";\n\n                    List&lt;String&gt; keys = Collections.singletonList(couponTemplateCacheKey);\n                    List&lt;String&gt; args = new ArrayList&lt;&gt;(actualCacheTargetMap.size() * 2 + 1);\n                    actualCacheTargetMap.forEach((key, value) -&gt; {\n                        args.add(key);\n                        args.add(value);\n                    });\n\n                    \/\/ \u4f18\u60e0\u5238\u6d3b\u52a8\u8fc7\u671f\u65f6\u95f4\u8f6c\u6362\u4e3a\u79d2\u7ea7\u522b\u7684 Unix \u65f6\u95f4\u6233\n                    args.add(String.valueOf(couponTemplateDO.getValidEndTime().getTime() \/ 1000));\n\n                    \/\/ \u6267\u884c LUA \u811a\u672c\n                    stringRedisTemplate.execute(\n                            new DefaultRedisScript&lt;&gt;(luaScript, Long.class),\n                            keys,\n                            args.toArray()\n                    );\n                    couponTemplateCacheMap = cacheTargetMap.entrySet()\n                            .stream()\n                            .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));\n                }\n            } finally {\n                lock.unlock();\n            }\n        }\n\n        return BeanUtil.mapToBean(couponTemplateCacheMap, CouponTemplateQueryRespDTO.class, false, CopyOptions.create());\n    }\n}\n<\/code><\/pre>\n\n\n\n<pre class=\"wp-block-code\"><code>public final class EngineRedisConstant {\n\n    \/**\n     * \u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58 Key\n     *\/\n    public static final String COUPON_TEMPLATE_KEY = \"one-coupon_engine:template:%s\";\n\n    \/**\n     * \u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u5206\u5e03\u5f0f\u9501 Key\n     *\/\n    public static final String LOCK_COUPON_TEMPLATE_KEY = \"one-coupon_engine:lock:template:%s\";\n}\n<\/code><\/pre>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"863\" height=\"894\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203693-image.png\" alt=\"\" class=\"wp-image-1297\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203693-image.png 863w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203693-image-290x300.png 290w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203693-image-768x796.png 768w\" sizes=\"auto, (max-width: 863px) 100vw, 863px\" \/><\/figure>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"722\" height=\"483\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203709-\u5c4f\u5e55\u622a\u56fe-2025-09-18-214530.png\" alt=\"\" class=\"wp-image-1298\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203709-\u5c4f\u5e55\u622a\u56fe-2025-09-18-214530.png 722w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758203709-\u5c4f\u5e55\u622a\u56fe-2025-09-18-214530-300x201.png 300w\" sizes=\"auto, (max-width: 722px) 100vw, 722px\" \/><\/figure>\n\n\n\n<h2 class=\"wp-block-heading\">\u7b2c17\u5c0f\u8282\uff1a\u7528\u6237\u67e5\u8be2\u4f18\u60e0\u5238\u4e4b\u7f13\u5b58\u7a7f\u900f<\/h2>\n\n\n\n<h2 class=\"wp-block-heading\">\u4e1a\u52a1\u80cc\u666f<\/h2>\n\n\n\n<p>\u5728\u4e0a\u4e00\u8282\u4e2d\uff0c\u6211\u4eec\u8ba8\u8bba\u4e86\u6b63\u5e38\u7528\u6237\u5728\u8bbf\u95ee\u4f18\u60e0\u5238\u65f6\u53ef\u80fd\u9047\u5230\u7684\u7f13\u5b58\u51fb\u7a7f\u95ee\u9898\uff0c\u5e76\u4ecb\u7ecd\u4e86\u7f13\u5b58\u9884\u70ed\u3001\u7f13\u5b58\u6c38\u4e0d\u8fc7\u671f\u3001\u5206\u5e03\u5f0f\u9501\u3001\u53cc\u91cd\u5224\u5b9a\u9501\u3001\u5206\u7247\u5206\u5e03\u5f0f\u9501\u7b49\u6280\u672f\u6765\u5e94\u5bf9\u8fd9\u4e9b\u95ee\u9898\u3002\u7136\u800c\uff0c\u8fd8\u6709\u4e00\u4e2a\u95ee\u9898\u9700\u8981\u89e3\u51b3\uff1a\u5982\u679c\u7528\u6237\u9891\u7e41\u8bbf\u95ee\u6570\u636e\u5e93\u4e2d\u4e0d\u5b58\u5728\u7684\u6570\u636e\uff0c\u5c31\u65e0\u6cd5\u6709\u6548\u4f7f\u7528\u7f13\u5b58\uff0c\u6bcf\u6b21\u90fd\u9700\u8981\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u8fd9\u5c06\u5bfc\u81f4\u6570\u636e\u5e93\u627f\u53d7\u8f83\u5927\u7684\u538b\u529b\u3002\u8fd9\u4e5f\u5c31\u662f\u7f13\u5b58\u7a7f\u900f\u95ee\u9898\u3002<\/p>\n\n\n\n<p>\u7f13\u5b58\u7a7f\u900f\u95ee\u9898\u65f6\u5e8f\u56fe\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"636\" width=\"550\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/59830d1da733c00890702f5621eb48e5.png\" alt=\"\"><\/p>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"git\">Git \u5206\u652f<\/h2>\n\n\n\n<p>20240827_dev_coupon-template-querypv-v2_cache_ding.ma<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4ec0\u4e48\u662f\u7f13\u5b58\u7a7f\u900f\uff1f<\/h2>\n\n\n\n<p>\u7f13\u5b58\u7a7f\u900f\u662f\u6307\u7531\u4e8e<strong>\u8bf7\u6c42\u6ca1\u6709\u529e\u6cd5\u547d\u4e2d\u7f13\u5b58<\/strong>\uff0c\u56e0\u6b64\u5c31\u4f1a\u76f4\u63a5\u6253\u5230\u6570\u636e\u5e93\uff0c\u5f53\u8bf7\u6c42\u91cf\u8f83\u5927\u65f6\uff0c\u5927\u91cf\u7684\u8bf7\u6c42\u5c31\u53ef\u80fd\u4f1a\u76f4\u63a5\u628a\u6570\u636e\u5e93\u6253\u6302\u3002<\/p>\n\n\n\n<p>\u901a\u5e38\u60c5\u51b5\u4e0b\uff0c\u7f13\u5b58\u662f<strong>\u4e3a\u4e86\u63d0\u9ad8\u6570\u636e\u8bbf\u95ee\u901f\u5ea6\uff0c\u907f\u514d\u9891\u7e41\u67e5\u8be2\u6570\u636e\u5e93<\/strong>\u3002\u4f46\u5982\u679c\u653b\u51fb\u8005\u6545\u610f\u8bf7\u6c42\u7f13\u5b58\u4e2d\u4e0d\u5b58\u5728\u7684\u6570\u636e\uff0c\u5c31\u4f1a\u5bfc\u81f4\u7f13\u5b58\u4e0d\u547d\u4e2d\uff0c\u8bf7\u6c42\u76f4\u63a5\u8bbf\u95ee\u6570\u636e\u5e93\u3002<\/p>\n\n\n\n<p>\u6ca1\u6709\u7ecf\u8fc7\u7f13\u5b58\u7a7f\u900f\u5904\u7406\u7684\u4e1a\u52a1\u4f2a\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectUser(String userId) {\n    String cacheData = cache.get(userId);\n    if (StrUtil.isBlank(cacheData)) {\n        String dbData = userMapper.selectId(userId);\n        if (StrUtil.isNotBlank(dbData)) {\n            cahce.set(userId, dbData);\n            cacheData = dbData;\n        } else {\n            throw new RuntimeException();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<h2 class=\"wp-block-heading\">\u7f13\u5b58\u7a7f\u900f\u5e38\u89c1\u89e3\u51b3\u65b9\u6848<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u7a7a\u5bf9\u8c61\u7f13\u5b58<\/h3>\n\n\n\n<p><strong>\u5f53\u67e5\u8be2\u7ed3\u679c\u4e3a\u7a7a\u65f6\uff0c\u4e5f\u5c06\u7ed3\u679c\u8fdb\u884c\u7f13\u5b58<\/strong>\uff0c\u4f46\u662f\u8bbe\u7f6e\u4e00\u4e2a\u8f83\u77ed\u7684\u8fc7\u671f\u65f6\u95f4\u3002\u8fd9\u6837\u5728\u63a5\u4e0b\u6765\u7684\u4e00\u6bb5\u65f6\u95f4\u5185\uff0c\u5982\u679c\u518d\u6b21\u8bf7\u6c42\u76f8\u540c\u7684\u6570\u636e\uff0c\u5c31\u53ef\u4ee5\u76f4\u63a5\u4ece\u7f13\u5b58\u4e2d\u83b7\u53d6\uff0c\u800c\u4e0d\u662f\u518d\u6b21\u8bbf\u95ee\u6570\u636e\u5e93\uff0c\u53ef\u4ee5\u4e00\u5b9a\u7a0b\u5ea6\u4e0a\u89e3\u51b3\u7f13\u5b58\u7a7f\u900f\u95ee\u9898\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"724\" width=\"588\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/34107453aa6c24412dff698037a51a5d.png\" alt=\"\"><\/p>\n\n\n\n<p>\u7f13\u5b58\u7a7a\u503c\u903b\u8f91\u4f2a\u4ee3\u7801\u5b9e\u73b0\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectUser(String userId) {\n    String cacheData = cache.get(userId);\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u5224\u65ad Key \u662f\u5426\u5305\u542b\u7a7a\u503c\u7f13\u5b58\uff0c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u7ee7\u7eed\u6d41\u7a0b\n        Boolean cacheIsNull = cache.hasKey(\"is-null_\" + userId);\n        if (cacheIsNull) {\n            throw new RuntimeException();\n        }\n        String dbData = userMapper.selectId(userId);\n        if (StrUtil.isNotBlank(dbData)) {\n            cahce.set(userId, dbData);\n            cacheData = dbData;\n        } else {\n            \/\/ \u67e5\u8be2\u6570\u636e\u5e93\u4e2d\u4e0d\u5b58\u5728\u6570\u636e\uff0c\u6dfb\u52a0\u7a7a\u503c\u7f13\u5b58\u5e76\u8fd4\u56de\n            cache.set(\"is-null_\" + userId, \u8f83\u77ed\u8fc7\u671f\u65f6\u95f4);\n            throw new RuntimeException();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u8fd9\u79cd\u65b9\u5f0f\u662f\u6bd4\u8f83\u7b80\u5355\u7684\u4e00\u79cd\u5b9e\u73b0\u65b9\u6848\uff0c\u4f1a\u5b58\u5728\u4e00\u4e9b\u5f0a\u7aef\u3002\u90a3\u5c31\u662f\u5f53<strong>\u77ed\u65f6\u95f4\u5185\u5b58\u5728\u5927\u91cf\u6076\u610f\u8bf7\u6c42\uff0c\u7f13\u5b58\u7cfb\u7edf\u4f1a\u5b58\u5728\u5927\u91cf\u7684\u5185\u5b58\u5360\u7528<\/strong>\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u5e03\u9686\u8fc7\u6ee4\u5668<\/h3>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-1\">2.1 \u4ec0\u4e48\u662f\u5e03\u9686\u8fc7\u6ee4\u5668<\/h4>\n\n\n\n<p>\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u4e00\u79cd\u6570\u636e\u7ed3\u6784\uff0c\u7528\u4e8e\u5feb\u901f\u5224\u65ad\u4e00\u4e2a\u5143\u7d20\u662f\u5426\u5b58\u5728\u4e8e\u4e00\u4e2a\u96c6\u5408\u4e2d\u3002\u5b83\u4ee5\u727a\u7272\u4e00\u5b9a\u7684\u51c6\u786e\u6027\u4e3a\u4ee3\u4ef7\uff0c\u6362\u53d6\u4e86\u5b58\u50a8\u7a7a\u95f4\u7684\u6781\u5927\u8282\u7701\u548c\u67e5\u8be2\u901f\u5ea6\u7684\u663e\u8457\u63d0\u5347\u3002<\/p>\n\n\n\n<p>\u5177\u4f53\u6765\u8bf4\uff0c\u5e03\u9686\u8fc7\u6ee4\u5668\u5305\u542b\u4e00\u4e2a\u4f4d\u6570\u7ec4\u548c\u4e00\u7ec4\u54c8\u5e0c\u51fd\u6570\u3002\u4f4d\u6570\u7ec4\u7684\u521d\u59cb\u503c\u5168\u90e8\u7f6e\u4e3a 0\u3002\u5728\u63d2\u5165\u4e00\u4e2a\u5143\u7d20\u65f6\uff0c\u5c06\u8be5\u5143\u7d20\u7ecf\u8fc7\u591a\u4e2a\u54c8\u5e0c\u51fd\u6570\u6620\u5c04\u5230\u4f4d\u6570\u7ec4\u4e0a\u7684\u591a\u4e2a\u4f4d\u7f6e\uff0c\u5e76\u5c06\u8fd9\u4e9b\u4f4d\u7f6e\u7684\u503c\u7f6e\u4e3a 1\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"1142\" width=\"2000\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/6e97a26ac070458a0d07fb5ccf5cb6be.png\" alt=\"\"><\/p>\n\n\n\n<p>\u56e0\u4e3a\u6bcf\u4e2a\u5143\u7d20\u5b58\u50a8\u90fd\u662f\u4ee5\u4f4d\u6765\u5b58\u50a8\uff0c\u800c\u4e0d\u662f\u5b57\u8282\uff0c\u6240\u4ee5\u5143\u7d20\u7684\u5360\u7528\u7a7a\u95f4\u975e\u5e38\u5c0f\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>1 \u5b57\u8282\uff08Byte\uff09= 8 \u4f4d\uff08Bit\uff09\u5728\u8ba1\u7b97\u673a\u79d1\u5b66\u4e2d\uff0c\u6570\u636e\u5b58\u50a8\u7684\u6700\u5c0f\u5355\u4f4d\u662f\u4f4d\uff08Bit\uff09\uff0c\u800c\u5b57\u8282\uff08Byte\uff09\u5219\u662f\u4e00\u4e2a\u5e38\u7528\u7684\u6570\u636e\u5b58\u50a8\u5355\u4f4d\uff0c\u901a\u5e38\u75318\u4e2a\u4f4d\u7ec4\u6210\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<p>\u5728\u67e5\u8be2\u4e00\u4e2a\u5143\u7d20\u662f\u5426\u5b58\u5728\u65f6\uff0c\u4f1a\u5c06\u8be5\u5143\u7d20\u7ecf\u8fc7\u591a\u4e2a\u54c8\u5e0c\u51fd\u6570\u6620\u5c04\u5230\u4f4d\u6570\u7ec4\u4e0a\u7684\u591a\u4e2a\u4f4d\u7f6e\uff0c\u5982\u679c\u6240\u6709\u4f4d\u7f6e\u7684\u503c\u90fd\u4e3a 1\uff0c\u5219\u8ba4\u4e3a\u5143\u7d20\u5b58\u5728\uff1b\u5982\u679c\u5b58\u5728\u4efb\u4e00\u4f4d\u7f6e\u7684\u503c\u4e3a 0\uff0c\u5219\u8ba4\u4e3a\u5143\u7d20\u4e0d\u5b58\u5728\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-2\">2.2 \u5e03\u9686\u8fc7\u6ee4\u5668\u4f18\u7f3a\u70b9<\/h4>\n\n\n\n<p>\u5e03\u9686\u8fc7\u6ee4\u5668\u7684\u4f18\u70b9\u5728\u4e8e\u5b83\u53ef\u4ee5\u9ad8\u6548\u5730\u5224\u65ad\u4e00\u4e2a\u5143\u7d20\u662f\u5426\u5c5e\u4e8e\u4e00\u4e2a\u5927\u89c4\u6a21\u96c6\u5408\uff0c\u4e14\u5177\u6709\u6781\u4f4e\u7684\u5b58\u50a8\u7a7a\u95f4\u8981\u6c42\u3002\u5982\u679c\u5b58\u50a8 1\u4ebf\u5143\u7d20\uff0c\u8bef\u5224\u7387\u8bbe\u7f6e\u4e3a 0.001 \u4e5f\u5c31\u662f\u5343\u5206\u4e4b\u4e00\uff0c\u4ec5\u9700\u8981\u5360\u7528 171M \u5de6\u53f3\u7684\u5185\u5b58\u3002<\/p>\n\n\n\n<p>\u7f3a\u70b9\u5728\u4e8e\u53ef\u80fd\u4f1a\u5b58\u5728\u4e00\u5b9a\u7684\u8bef\u5224\u7387\u3002<\/p>\n\n\n\n<p>\u5b83\u5728\u5b9e\u9645\u5e94\u7528\u4e2d\u5e38\u7528\u4e8e\u7f13\u5b58\u573a\u666f\u4e0b\u7f13\u5b58\u7a7f\u900f\u95ee\u9898\uff0c\u5bf9\u8bbf\u95ee\u8bf7\u6c42\u505a\u4e00\u4e2a\u5feb\u901f\u5224\u65ad\u673a\u5236\u3002\u4f7f\u7528\u5e03\u9686\u8fc7\u6ee4\u5668\u80fd\u591f\u6709\u6548\u51cf\u8f7b\u5bf9\u5e95\u5c42\u5b58\u50a8\u7cfb\u7edf\u7684\u8bbf\u95ee\u4ee5\u53ca\u7f13\u5b58\u7cfb\u7edf\u7684\u5b58\u50a8\u538b\u529b\u3002<\/p>\n\n\n\n<p>\u4f46\u662f\u5e03\u9686\u8fc7\u6ee4\u5668\u672c\u8eab\u4e5f\u5b58\u5728\u4e00\u4e9b\u201c\u5f0a\u7aef\u201d\uff0c\u90a3\u5c31\u662f\u4e0d\u652f\u6301\u5220\u9664\u5143\u7d20\u3002\u56e0\u4e3a\u5b83\u662f\u4e00\u79cd\u57fa\u4e8e\u54c8\u5e0c\u7684\u6570\u636e\u7ed3\u6784\uff0c\u5220\u9664\u5143\u7d20\u4f1a\u6d89\u53ca\u5230\u591a\u4e2a\u54c8\u5e0c\u51fd\u6570\u4e4b\u95f4\u7684\u51b2\u7a81\u95ee\u9898\uff0c\u8fd9\u6837\u4f1a\u5bfc\u81f4\u5220\u9664\u4e00\u4e2a\u5143\u7d20\u53ef\u80fd\u4f1a\u5f71\u54cd\u5230\u5176\u4ed6\u5143\u7d20\u7684\u6b63\u786e\u6027\u3002<\/p>\n\n\n\n<p>\u603b\u7684\u6765\u8bf4\uff0c\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u4e00\u79cd\u975e\u5e38\u9ad8\u6548\u7684\u6570\u636e\u7ed3\u6784\uff0c\u9002\u7528\u4e8e\u90a3\u4e9b\u53ef\u4ee5\u5bb9\u5fcd\u4e00\u5b9a\u7684\u8bef\u5224\u7387\u7684\u573a\u5408\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-3\">2.3 \u5e03\u9686\u8fc7\u6ee4\u5668\u89e3\u51b3\u7f13\u5b58\u7a7f\u900f<\/h4>\n\n\n\n<p>\u53ef\u4ee5\u5c06\u6240\u6709\u5b58\u91cf\u6570\u636e\u5168\u90e8\u653e\u5165\u5e03\u9686\u8fc7\u6ee4\u5668\uff0c\u7136\u540e\u5982\u679c\u7f13\u5b58\u4e2d\u4e0d\u5b58\u5728\u6570\u636e\uff0c\u7d27\u63a5\u7740\u5224\u65ad\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u5426\u5b58\u5728\uff0c\u5982\u679c\u5b58\u5728\u8bbf\u95ee\u6570\u636e\u5e93\u8bf7\u6c42\u6570\u636e\uff0c\u5982\u679c\u4e0d\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\u9519\u8bef\u54cd\u5e94\u5373\u53ef\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"800\" width=\"765\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/bfb11ac2d74c0918ad0b2687176836f9.png\" alt=\"\"><\/p>\n\n\n\n<p>\u4f7f\u7528\u5e03\u9686\u8fc7\u6ee4\u5668\u89e3\u51b3\u7f13\u5b58\u7a7f\u900f\u4f2a\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectUser(String userId) {\n    String cacheData = cache.get(userId);\n    if (StrUtil.isBlank(cacheData)) {\n        if (!bloomFilter.contains(fullShortUrl)) {\n            throw new RuntimeException();\n        }\n        String dbData = userMapper.selectId(userId);\n        if (StrUtil.isNotBlank(dbData)) {\n            cahce.set(userId, dbData);\n            cacheData = dbData;\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u4f46\u662f\u8fd9\u79cd\u95ee\u9898\u8fd8\u662f\u4f1a\u6709\u4e00\u4e9b\u5c0f\u6982\u7387\u95ee\u9898\uff0c\u90a3\u5c31\u662f\u5982\u679c\u4f7f\u7528\u4e00\u79cd<strong>\u5c0f\u6982\u7387\u8bef\u5224\u7684\u7f13\u5b58\u8fdb\u884c\u653b\u51fb<\/strong>\uff0c\u4f9d\u7136\u4f1a\u5bf9\u6570\u636e\u5e93\u9020\u6210\u6bd4\u8f83\u5927\u7684\u538b\u529b\u3002\u8fd9\u4e2a\u600e\u4e48\u7406\u89e3\u5462\uff1f<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u6bd4\u5982\u8bf4\u4e00\u4e2a\u4f18\u60e0\u5238 ID \u662f 1827975299049058306\uff0c\u6211\u901a\u8fc7\u4f18\u60e0\u5238 ID \u89c4\u5219\uff0c\u6a21\u62df\u4e00\u4e2a\u4e0d\u5b58\u5728\u7684\u4f46\u5f88\u76f8\u8fd1\u7684\uff0c\u6bd4\u5982 1827975299049058307\uff0c\u53bb\u78b0\u649e\u90a3\u4e2a\u8bef\u5224\u7684\u6982\u7387\uff1b<\/li>\n\n\n\n<li>2. \u600e\u4e48\u5224\u65ad\u8fd9\u4e2a\u6570\u636e\u662f\u4e0d\u662f\u5b58\u5728\uff1f\u5c31\u662f\u770b\u63a5\u53e3\u7684\u54cd\u5e94\u65f6\u95f4\uff0c\u76f4\u63a5\u67e5\u8be2\u7f13\u5b58\u548c\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u7edd\u5bf9\u7684\u6beb\u79d2\u7ea7\uff0c\u6bd4\u5982 5 \u6beb\u79d2\uff0c\u800c\u4e14\u6027\u80fd\u57fa\u672c\u4e0a\u6bd4\u8f83\u6052\u5b9a\u3002\u90a3\u6211\u4eec\u5c31\u53ef\u4ee5\u6839\u636e\u76f8\u5e94\u65f6\u95f4\u662f\u5426\u5927\u4e8e 5 \u6beb\u79d2\uff0c\u56e0\u4e3a\u8bef\u5224\u4e86\u8fd8\u4f1a\u67e5\u4e00\u6b21\u6570\u636e\u5e93\uff1b<\/li>\n\n\n\n<li>3. \u5982\u679c\u67e5\u8be2\u7b2c\u4e00\u6b21\u5927\u4e8e 5 \u6beb\u79d2\u4e14\u6570\u636e\u8fd4\u56de\u4e3a\u7a7a\uff0c\u90a3\u5c31\u8bc1\u660e\u8fd9\u662f\u4e2a\u78b0\u649e\u6f0f\u7f51\u4e4b\u9c7c\uff0c\u76f4\u63a5\u62ff\u9ad8\u5e76\u53d1\u8bbf\u95ee\u5373\u53ef\uff0c\u8fd8\u662f\u4f1a\u8bf7\u6c42\u5230\u6570\u636e\u5e93\u3002<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u5e03\u9686\u8fc7\u6ee4\u5668+\u7a7a\u503c\u7f13\u5b58+\u5206\u5e03\u5f0f\u9501<\/h3>\n\n\n\n<p>\u5982\u679c\u8bf4\u7f13\u5b58\u4e0d\u5b58\u5728\uff0c\u90a3\u4e48\u5c31\u901a\u8fc7\u5e03\u9686\u8fc7\u6ee4\u5668\u8fdb\u884c\u521d\u6b65\u7b5b\u9009\uff0c\u7136\u540e<strong>\u5224\u65ad\u662f\u5426\u5b58\u5728\u7f13\u5b58\u7a7a\u503c\uff0c\u5982\u679c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\u5931\u8d25<\/strong>\u3002\u5982\u679c\u4e0d\u5b58\u5728\u7f13\u5b58\u7a7a\u503c\uff0c\u4f7f\u7528\u9501\u673a\u5236\u907f\u514d\u591a\u4e2a\u76f8\u540c\u8bf7\u6c42\u540c\u65f6\u8bbf\u95ee\u6570\u636e\u5e93\u3002\u6700\u540e\uff0c<strong>\u5982\u679c\u8bf7\u6c42\u6570\u636e\u5e93\u4e3a\u7a7a\uff0c\u90a3\u4e48\u5c06\u4e3a\u7a7a\u7684 Key \u8fdb\u884c\u7a7a\u5bf9\u8c61\u503c\u7f13\u5b58<\/strong>\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"1186\" width=\"946\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/d8c6667fdf75b3ca118528873007016e.png\" alt=\"\"><\/p>\n\n\n\n<p>\u591a\u91cd\u65b9\u6848\u4f2a\u4ee3\u7801\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectUser(String userId) {\n    String cacheData = cache.get(userId);\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u5224\u65ad Key \u662f\u5426\u5b58\u5728\u5e03\u9686\u8fc7\u6ee4\u5668\uff0c\u5b58\u5728\u5219\u7ee7\u7eed\u6d41\u7a0b\uff0c\u5426\u5219\u76f4\u63a5\u8fd4\u56de\n        if (!bloomFilter.contains(fullShortUrl)) {\n            throw new RuntimeException();\n        }\n\u200b\n        \/\/ \u5224\u65ad Key \u662f\u5426\u5305\u542b\u7a7a\u503c\u7f13\u5b58\uff0c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u7ee7\u7eed\u6d41\u7a0b\n        Boolean cacheIsNull = cache.hasKey(\"is-null_\" + userId);\n        if (cacheIsNull) {\n            throw new RuntimeException();\n        }\n\u200b\n        \/\/ \u83b7\u53d6\u5206\u5e03\u5f0f\u9501\n        Lock lock = getLock(userId);\n        lock.lock();\n\u200b\n        try {\n            \/\/ \u62ff\u5230\u9501\u4e4b\u540e\u8fdb\u884c\u53cc\u91cd\u5224\u5b9a\uff0c\u5982\u679c\u7f13\u5b58\u5df2\u7ecf\u5b58\u5728\u5219\u76f4\u63a5\u8fd4\u56de\u5373\u53ef\n            cacheData = cache.get(userId);\n            if (StrUtil.isBlank(cacheData)) {\n                String dbData = userMapper.selectId(userId);\n                if (StrUtil.isNotBlank(dbData)) {\n                    cahce.set(userId, dbData);\n                    cacheData = dbData;\n                } else {\n                    \/\/ \u67e5\u8be2\u6570\u636e\u5e93\u4e2d\u4e0d\u5b58\u5728\u6570\u636e\uff0c\u6dfb\u52a0\u7a7a\u503c\u7f13\u5b58\u5e76\u8fd4\u56de\n                    cache.set(\"is-null_\" + userId, \u8f83\u77ed\u8fc7\u671f\u65f6\u95f4);\n                    throw new RuntimeException();\n                }\n            }\n        } finally {\n            lock.unlock();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u4e4b\u524d\u6211\u8003\u8651\u5230\u8fd9\u5c31\u7ed3\u675f\u4e86\uff0c\u5f97\u76ca\u4e8e\u6211\u5f00\u6e90\u7684 SaaS \u77ed\u94fe\u63a5\u9879\u76ee\uff0c\u6709\u5f88\u591a\u7ec6\u5fc3\u7684\u540c\u5b66\u7ed9\u6211\u63d0\u4e86\u4e00\u4e9b\u5efa\u8bae\u3002\u90a3\u5c31\u662f\u5728\u83b7\u53d6\u5230\u9501\u540e\uff0c<strong>\u4e0d\u6b62\u5bf9\u6b63\u5e38\u7f13\u5b58\u53cc\u91cd\u5224\u5b9a\uff0c\u540c\u65f6\u4e5f\u8981\u5bf9\u7a7a\u503c\u7f13\u5b58\u5bf9\u8c61\u505a\u53cc\u91cd\u5224\u5b9a<\/strong>\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public String selectUser(String userId) {\n    String cacheData = cache.get(userId);\n    if (StrUtil.isBlank(cacheData)) {\n        \/\/ \u5224\u65ad Key \u662f\u5426\u5b58\u5728\u5e03\u9686\u8fc7\u6ee4\u5668\uff0c\u5b58\u5728\u5219\u7ee7\u7eed\u6d41\u7a0b\uff0c\u5426\u5219\u76f4\u63a5\u8fd4\u56de\n        if (!bloomFilter.contains(fullShortUrl)) {\n            throw new RuntimeException();\n        }\n\u200b\n        \/\/ \u5224\u65ad Key \u662f\u5426\u5305\u542b\u7a7a\u503c\u7f13\u5b58\uff0c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u7ee7\u7eed\u6d41\u7a0b\n        Boolean cacheIsNull = cache.hasKey(\"is-null_\" + userId);\n        if (cacheIsNull) {\n            throw new RuntimeException();\n        }\n\u200b\n        \/\/ \u83b7\u53d6\u5206\u5e03\u5f0f\u9501\n        Lock lock = getLock(userId);\n        lock.lock();\n\u200b\n        try {\n            \/\/ \u62ff\u5230\u9501\u4e4b\u540e\u8fdb\u884c\u53cc\u91cd\u5224\u5b9a\uff0c\u5982\u679c\u7f13\u5b58\u5df2\u7ecf\u5b58\u5728\u5219\u76f4\u63a5\u8fd4\u56de\u5373\u53ef\n            cacheData = cache.get(userId);\n            if (StrUtil.isNotBlank(cacheData)) {\n                return cacheData;\n            }\n\u200b\n            \/\/ \u62ff\u5230\u9501\u4e4b\u540e\u8fdb\u884c\u53cc\u91cd\u5224\u5b9a\uff0c\u5982\u679c\u7a7a\u503c\u7f13\u5b58\u5df2\u7ecf\u5b58\u5728\u5219\u76f4\u63a5\u7ec8\u6b62\u6d41\u7a0b\u5373\u53ef\n            cacheIsNull = cache.hasKey(\"is-null_\" + userId);\n            if (!cacheIsNull) {\n                throw new RuntimeException();\n            }\n\u200b\n            \/\/ \u6839\u636e\u7528\u6237\u6807\u8bc6\u67e5\u8be2\u6570\u636e\u5e93\u8bb0\u5f55\n            String dbData = userMapper.selectId(userId);\n            if (StrUtil.isNotBlank(dbData)) {\n                cahce.set(userId, dbData);\n                cacheData = dbData;\n            } else {\n                \/\/ \u67e5\u8be2\u6570\u636e\u5e93\u4e2d\u4e0d\u5b58\u5728\u6570\u636e\uff0c\u6dfb\u52a0\u7a7a\u503c\u7f13\u5b58\u5e76\u8fd4\u56de\n                cache.set(\"is-null_\" + userId, \u8f83\u77ed\u8fc7\u671f\u65f6\u95f4);\n                throw new RuntimeException();\n            }\n        } finally {\n            lock.unlock();\n        }\n    }\n    return cacheData;\n}<\/code><\/pre>\n\n\n\n<p>\u81f3\u6b64\uff0c\u7f13\u5b58\u7a7f\u900f\u89e3\u51b3\u65b9\u6848\u5b8c\u7f8e\u7ed3\u675f\u3002<\/p>\n\n\n\n<p>\u4e0a\u9762\u5199\u5f97\u7f13\u5b58\u7a7f\u900f\u4ee3\u7801\u601d\u8def\uff0c\u53ea\u8981\u4e0d\u51fa\u73b0\u6781\u7aef\u573a\u666f\uff0c\u5927\u6982\u7387\u80fd\u6db5\u76d6 90% \u5de5\u4f5c\u5f53\u4e2d\u7684\u4e1a\u52a1\u573a\u666f\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4f18\u60e0\u5238\u6a21\u677f\u5f15\u5165\u7f13\u5b58\u7a7f\u900f\u89e3\u51b3\u65b9\u6848<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u521b\u5efa\u5e03\u9686\u8fc7\u6ee4\u5668<\/h3>\n\n\n\n<p>\u4e0b\u9762\u8fd9\u4e2a\u914d\u7f6e\u9700\u8981\u5728\u4f18\u60e0\u5238\u540e\u7ba1\u548c\u5f15\u64ce\u6a21\u5757<strong>\u90fd\u9700\u8981\u6dfb\u52a0<\/strong>\uff0c\u4ee5\u540e\u7ba1\u4e2d\u914d\u7f6e\u4ee3\u7801\u4e3e\u4f8b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Configuration\npublic class RBloomFilterConfiguration {\n\n    \/**\n     * \u4f18\u60e0\u5238\u67e5\u8be2\u7f13\u5b58\u7a7f\u900f\u5e03\u9686\u8fc7\u6ee4\u5668\n     *\/\n    @Bean\n    public RBloomFilter&lt;String&gt; couponTemplateQueryBloomFilter(RedissonClient redissonClient) {\n        RBloomFilter&lt;String&gt; bloomFilter = redissonClient.getBloomFilter(\"couponTemplateQueryBloomFilter\");\n        bloomFilter.tryInit(640L, 0.001);\n        return bloomFilter;\n    }\n}\n<\/code><\/pre>\n\n\n\n<p>\u5176\u4e2d <code>tryInit<\/code> \u6709\u4e24\u4e2a\u53c2\u6570\uff0c\u4ee3\u8868\u9884\u4f30\u63d2\u5165\u91cf\u548c\u9519\u8bef\u6982\u7387\uff0c\u56e0\u4e3a\u4f1a\u6839\u636e\u8fd9\u4e24\u4e2a\u53c2\u6570\u8fdb\u884c\u521d\u59cb\u5316\u5e03\u9686\u8fc7\u6ee4\u5668\u7684\u4f4d\u6570\u7ec4\uff0c\u4e0d\u5efa\u8bae\u5927\u5bb6\u8bbe\u7f6e\u592a\u5927\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-1-expected-insertions\">2.1 \u9884\u8ba1\u63d2\u5165\u91cf expectedInsertions<\/h4>\n\n\n\n<p>\u8fd9\u662f\u4e00\u4e2a\u4f30\u8ba1\u503c\uff0c\u8868\u793a\u5e03\u9686\u8fc7\u6ee4\u5668\u9884\u671f\u5c06\u4f1a\u63d2\u5165\u7684\u5143\u7d20\u603b\u6570\u3002<\/p>\n\n\n\n<p>\u901a\u8fc7\u77e5\u9053\u9884\u671f\u7684\u63d2\u5165\u91cf\uff0c\u5e03\u9686\u8fc7\u6ee4\u5668\u53ef\u4ee5\u6839\u636e\u8fd9\u4e2a\u4f30\u8ba1\u503c\u6765\u5408\u7406\u5730\u914d\u7f6e\u4f4d\u6570\u7ec4\u7684\u5927\u5c0f\u548c\u54c8\u5e0c\u51fd\u6570\u7684\u6570\u91cf\u3002\u8fd9\u6837\u53ef\u4ee5\u5728\u8fbe\u5230\u6307\u5b9a\u9519\u8bef\u7387\u7684\u60c5\u51b5\u4e0b\uff0c\u6700\u5927\u9650\u5ea6\u5730\u8282\u7701\u5185\u5b58\u3002\u63d2\u5165\u7684\u5143\u7d20\u6570\u91cf\u5982\u679c\u8d85\u51fa\u9884\u671f\uff0c\u4f1a\u589e\u52a0\u8bef\u5224\u7684\u6982\u7387\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-2-false-probability\">2.2 \u9519\u8bef\u6982\u7387 falseProbability<\/h4>\n\n\n\n<p>\u8868\u793a\u5e03\u9686\u8fc7\u6ee4\u5668\u5728\u7ed9\u5b9a\u6761\u4ef6\u4e0b\u53ef\u80fd\u8fd4\u56de\u9519\u8bef\u7ed3\u679c\u7684\u6982\u7387\u3002<\/p>\n\n\n\n<p>\u9519\u8bef\u6982\u7387\u7528\u4e8e\u5728\u5e03\u9686\u8fc7\u6ee4\u5668\u521d\u59cb\u5316\u65f6\u786e\u5b9a\u5176\u4f4d\u6570\u7ec4\u7684\u5927\u5c0f\u548c\u54c8\u5e0c\u51fd\u6570\u7684\u6570\u91cf\u3002\u8f83\u4f4e\u7684\u8bef\u5224\u7387\u610f\u5473\u7740\u9700\u8981\u66f4\u5927\u7684\u4f4d\u6570\u7ec4\u548c\u66f4\u591a\u7684\u54c8\u5e0c\u51fd\u6570\uff0c\u4ece\u800c\u5360\u7528\u66f4\u591a\u7684\u5185\u5b58\u548c\u8ba1\u7b97\u65f6\u95f4\u3002\u53cd\u4e4b\uff0c\u8f83\u9ad8\u7684\u8bef\u5224\u7387\u5219\u610f\u5473\u7740\u66f4\u5c11\u7684\u5185\u5b58\u5360\u7528\u548c\u8ba1\u7b97\u6210\u672c\uff0c\u4f46\u9519\u8bef\u5224\u5b9a\u7684\u6982\u7387\u4e5f\u4f1a\u589e\u52a0\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u521b\u5efa\u4f18\u60e0\u5238\u6a21\u677f\u6dfb\u52a0\u5e03\u9686\u8fc7\u6ee4\u5668<\/h3>\n\n\n\n<p>\u6211\u4eec\u521b\u5efa\u4f18\u60e0\u5238\u6a21\u677f\u65b9\u6cd5\u91cc\uff0c\u9700\u8981\u5c06\u4f18\u60e0\u5238\u6a21\u677f ID \u5b58\u4e00\u4efd\u5230\u5e03\u9686\u8fc7\u6ee4\u5668\u4e2d\uff0c\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Slf4j\n@Service\n@RequiredArgsConstructor\npublic class CouponTemplateServiceImpl extends ServiceImpl&lt;CouponTemplateMapper, CouponTemplateDO&gt; implements CouponTemplateService {\n\u200b\n    \/\/ ......\n    private final RBloomFilter&lt;String&gt; couponTemplateQueryBloomFilter;\n\u200b\n    @Override\n    public void createCouponTemplate(CouponTemplateSaveReqDTO requestParam) {\n        \/\/ ......\n        \/\/ \u6dfb\u52a0\u4f18\u60e0\u5238\u6a21\u677f ID \u5230\u5e03\u9686\u8fc7\u6ee4\u5668\n        couponTemplateQueryBloomFilter.add(String.valueOf(couponTemplateDO.getId()));\n    }\n}<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u6539\u9020\u4f18\u60e0\u5238\u6a21\u677f\u67e5\u8be2\u7f13\u5b58\u7a7f\u900f<\/h3>\n\n\n\n<p>\u6211\u4eec\u5148\u5199\u4e00\u4e2a\u5355\u72ec\u7684\u7f13\u5b58\u7a7f\u900f\u89e3\u51b3\u65b9\u6848\uff0c\u540e\u9762\u518d\u548c\u51fb\u7a7f\u903b\u8f91\u7ed3\u5408\u4e00\u8d77\u3002<\/p>\n\n\n\n<p>\u67e5\u8be2\u4f18\u60e0\u5238\u6a21\u677f\u8bf7\u6c42\u7b2c\u4e00\u6b65\uff0c\u5224\u65ad\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u5426\u5b58\u5728\u6307\u5b9a\u6a21\u677f ID\uff0c\u4e0d\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\u9519\u8bef\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Slf4j\n@Service\n@RequiredArgsConstructor\npublic class CouponTemplateServiceImpl extends ServiceImpl&lt;CouponTemplateMapper, CouponTemplateDO&gt; implements CouponTemplateService {\n\u200b\n    private final RBloomFilter&lt;String&gt; couponTemplateQueryBloomFilter;\n\u200b\n    @Override\n    public CouponTemplateQueryRespDTO findCouponTemplate(CouponTemplateQueryReqDTO requestParam) {\n        if (!couponTemplateQueryBloomFilter.contains(requestParam.getCouponTemplateId())) {\n            throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\");\n        }\n\u200b\n        LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n        CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\u200b\n        return BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n    }\n}<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"4\">4. \u6539\u9020\u7ec4\u5408\u65b9\u6848<\/h3>\n\n\n\n<p>\u6211\u4eec\u5728\u8fd9\u4e2a\u65b9\u6cd5\u91cc\uff0c\u5f15\u5165\u7f13\u5b58\u5e03\u9686\u8fc7\u6ee4\u5668\u3001\u7a7a\u503c\u4ee5\u53ca\u5206\u5e03\u5f0f\u9501\u903b\u8f91\uff0c\u5e94\u7528\u5230\u6211\u4eec\u7684\u4f18\u60e0\u5238\u6a21\u677f\u67e5\u8be2\u89e3\u51b3\u65b9\u6848\u4e2d\u3002<\/p>\n\n\n\n<p>\u4ee3\u7801\u770b\u7740\u633a\u591a\uff0c\u4f46\u662f\u90fd\u662f\u6309\u7167\u6211\u4eec\u4e0a\u9762\u8bb2\u7684\u5e03\u9686\u8fc7\u6ee4\u5668\u3001\u63a7\u5236\u7f13\u5b58\u3001\u5206\u5e03\u5f0f\u9501\u903b\u8f91\u4e00\u6b65\u6b65\u6765\u7684\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Slf4j\n@Service\n@RequiredArgsConstructor\npublic class CouponTemplateServiceImpl extends ServiceImpl&lt;CouponTemplateMapper, CouponTemplateDO&gt; implements CouponTemplateService {\n\n    private final CouponTemplateMapper couponTemplateMapper;\n    private final StringRedisTemplate stringRedisTemplate;\n    private final RedissonClient redissonClient;\n    private final RBloomFilter&lt;String&gt; couponTemplateQueryBloomFilter;\n\n    @Override\n    public CouponTemplateQueryRespDTO findCouponTemplate(CouponTemplateQueryReqDTO requestParam) {\n        \/\/ \u67e5\u8be2 Redis \u7f13\u5b58\u4e2d\u662f\u5426\u5b58\u5728\u4f18\u60e0\u5238\u6a21\u677f\u4fe1\u606f\n        String couponTemplateCacheKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId());\n        Map&lt;Object, Object&gt; couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n\n        \/\/ \u5982\u679c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u9700\u8981\u901a\u8fc7\u5e03\u9686\u8fc7\u6ee4\u5668\u3001\u7f13\u5b58\u7a7a\u503c\u4ee5\u53ca\u53cc\u91cd\u5224\u5b9a\u9501\u7684\u5f62\u5f0f\u8bfb\u53d6\u6570\u636e\u5e93\u4e2d\u7684\u8bb0\u5f55\n        if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n            \/\/ \u5224\u65ad\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u5426\u5b58\u5728\u6307\u5b9a\u6a21\u677f ID\uff0c\u4e0d\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\u9519\u8bef\n            if (!couponTemplateQueryBloomFilter.contains(requestParam.getCouponTemplateId())) {\n                throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\");\n            }\n\n            \/\/ \u67e5\u8be2 Redis \u7f13\u5b58\u4e2d\u662f\u5426\u5b58\u5728\u4f18\u60e0\u5238\u6a21\u677f\u7a7a\u503c\u4fe1\u606f\uff0c\u5982\u679c\u6709\u4ee3\u8868\u6a21\u677f\u4e0d\u5b58\u5728\uff0c\u76f4\u63a5\u8fd4\u56de\n            String couponTemplateIsNullCacheKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_IS_NULL_KEY, requestParam.getCouponTemplateId());\n            Boolean hasKeyFlag = stringRedisTemplate.hasKey(couponTemplateIsNullCacheKey);\n            if (hasKeyFlag) {\n                throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\");\n            }\n\n            \/\/ \u83b7\u53d6\u4f18\u60e0\u5238\u6a21\u677f\u5206\u5e03\u5f0f\u9501\n            \/\/ \u5173\u4e8e\u7f13\u5b58\u51fb\u7a7f\u66f4\u591a\u6ce8\u91ca\u4e8b\u9879\uff0c\u6b22\u8fce\u67e5\u770b\u6211\u7684B\u7ad9\u89c6\u9891\uff1ahttps:\/\/www.bilibili.com\/video\/BV1qz421z7vC\n            RLock lock = redissonClient.getLock(String.format(EngineRedisConstant.LOCK_COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId()));\n            lock.lock();\n\n            try {\n                \/\/ \u53cc\u91cd\u5224\u5b9a\u7a7a\u503c\u7f13\u5b58\u662f\u5426\u5b58\u5728\uff0c\u5b58\u5728\u5219\u7ee7\u7eed\u629b\u5f02\u5e38\n                hasKeyFlag = stringRedisTemplate.hasKey(couponTemplateIsNullCacheKey);\n                if (hasKeyFlag) {\n                    throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\");\n                }\n\n                \/\/ \u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u4f18\u5316\u5927\u91cf\u8bf7\u6c42\u65e0\u610f\u4e49\u67e5\u8be2\u6570\u636e\u5e93\n                couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n                if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n                    LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                            .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                            .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                            .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n                    CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\n                    \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u8005\u5df2\u8fc7\u671f\u52a0\u5165\u7a7a\u503c\u7f13\u5b58\uff0c\u5e76\u4e14\u629b\u51fa\u5f02\u5e38\n                    if (couponTemplateDO == null) {\n                        stringRedisTemplate.opsForValue().set(couponTemplateIsNullCacheKey, \"\", 30, TimeUnit.MINUTES);\n                        throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u5df2\u8fc7\u671f\");\n                    }\n\n                    \/\/ \u901a\u8fc7\u5c06\u6570\u636e\u5e93\u7684\u8bb0\u5f55\u5e8f\u5217\u5316\u6210 JSON \u5b57\u7b26\u4e32\u653e\u5165 Redis \u7f13\u5b58\n                    CouponTemplateQueryRespDTO actualRespDTO = BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n                    Map&lt;String, Object&gt; cacheTargetMap = BeanUtil.beanToMap(actualRespDTO, false, true);\n                    Map&lt;String, String&gt; actualCacheTargetMap = cacheTargetMap.entrySet().stream()\n                            .collect(Collectors.toMap(\n                                    Map.Entry::getKey,\n                                    entry -&gt; entry.getValue() != null ? entry.getValue().toString() : \"\"\n                            ));\n\n                    \/\/ \u901a\u8fc7 LUA \u811a\u672c\u6267\u884c\u8bbe\u7f6e Hash \u6570\u636e\u4ee5\u53ca\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\n                    String luaScript = \"redis.call('HMSET', KEYS&#91;1], unpack(ARGV, 1, #ARGV - 1)) \" +\n                            \"redis.call('EXPIREAT', KEYS&#91;1], ARGV&#91;#ARGV])\";\n\n                    List&lt;String&gt; keys = Collections.singletonList(couponTemplateCacheKey);\n                    List&lt;String&gt; args = new ArrayList&lt;&gt;(actualCacheTargetMap.size() * 2 + 1);\n                    actualCacheTargetMap.forEach((key, value) -&gt; {\n                        args.add(key);\n                        args.add(value);\n                    });\n\n                    \/\/ \u4f18\u60e0\u5238\u6d3b\u52a8\u8fc7\u671f\u65f6\u95f4\u8f6c\u6362\u4e3a\u79d2\u7ea7\u522b\u7684 Unix \u65f6\u95f4\u6233\n                    args.add(String.valueOf(couponTemplateDO.getValidEndTime().getTime() \/ 1000));\n\n                    \/\/ \u6267\u884c LUA \u811a\u672c\n                    stringRedisTemplate.execute(\n                            new DefaultRedisScript&lt;&gt;(luaScript, Long.class),\n                            keys,\n                            args.toArray()\n                    );\n                    couponTemplateCacheMap = cacheTargetMap.entrySet()\n                            .stream()\n                            .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));\n                }\n            } finally {\n                lock.unlock();\n            }\n        }\n\n        return BeanUtil.mapToBean(couponTemplateCacheMap, CouponTemplateQueryRespDTO.class, false, CopyOptions.create());\n    }\n\n    \/**\n     * \u7f13\u5b58\u51fb\u7a7f\u89e3\u51b3\u65b9\u6848\n     *\/\n    public CouponTemplateQueryRespDTO findCouponTemplateV1(CouponTemplateQueryReqDTO requestParam) {\n        \/\/ \u67e5\u8be2 Redis \u7f13\u5b58\u4e2d\u662f\u5426\u5b58\u5728\u4f18\u60e0\u5238\u6a21\u677f\u4fe1\u606f\n        String couponTemplateCacheKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId());\n        Map&lt;Object, Object&gt; couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n\n        \/\/ \u5982\u679c\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\uff0c\u4e0d\u5b58\u5728\u9700\u8981\u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u7684\u5f62\u5f0f\u8bfb\u53d6\u6570\u636e\u5e93\u4e2d\u7684\u8bb0\u5f55\n        if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n            \/\/ \u83b7\u53d6\u4f18\u60e0\u5238\u6a21\u677f\u5206\u5e03\u5f0f\u9501\n            \/\/ \u5173\u4e8e\u7f13\u5b58\u51fb\u7a7f\u66f4\u591a\u6ce8\u91ca\u4e8b\u9879\uff0c\u6b22\u8fce\u67e5\u770b\u6211\u7684B\u7ad9\u89c6\u9891\uff1ahttps:\/\/www.bilibili.com\/video\/BV1qz421z7vC\n            RLock lock = redissonClient.getLock(String.format(EngineRedisConstant.LOCK_COUPON_TEMPLATE_KEY, requestParam.getCouponTemplateId()));\n            lock.lock();\n\n            try {\n                \/\/ \u901a\u8fc7\u53cc\u91cd\u5224\u5b9a\u9501\u4f18\u5316\u5927\u91cf\u8bf7\u6c42\u65e0\u610f\u4e49\u67e5\u8be2\u6570\u636e\u5e93\n                couponTemplateCacheMap = stringRedisTemplate.opsForHash().entries(couponTemplateCacheKey);\n                if (MapUtil.isEmpty(couponTemplateCacheMap)) {\n                    LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                            .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                            .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                            .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n                    CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\n                    \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u8005\u5df2\u8fc7\u671f\u76f4\u63a5\u629b\u51fa\u5f02\u5e38\n                    if (couponTemplateDO == null) {\n                        throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\u6216\u5df2\u8fc7\u671f\");\n                    }\n\n                    \/\/ \u901a\u8fc7\u5c06\u6570\u636e\u5e93\u7684\u8bb0\u5f55\u5e8f\u5217\u5316\u6210 JSON \u5b57\u7b26\u4e32\u653e\u5165 Redis \u7f13\u5b58\n                    CouponTemplateQueryRespDTO actualRespDTO = BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n                    Map&lt;String, Object&gt; cacheTargetMap = BeanUtil.beanToMap(actualRespDTO, false, true);\n                    Map&lt;String, String&gt; actualCacheTargetMap = cacheTargetMap.entrySet().stream()\n                            .collect(Collectors.toMap(\n                                    Map.Entry::getKey,\n                                    entry -&gt; entry.getValue() != null ? entry.getValue().toString() : \"\"\n                            ));\n\n                    \/\/ \u901a\u8fc7 LUA \u811a\u672c\u6267\u884c\u8bbe\u7f6e Hash \u6570\u636e\u4ee5\u53ca\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4\n                    String luaScript = \"redis.call('HMSET', KEYS&#91;1], unpack(ARGV, 1, #ARGV - 1)) \" +\n                            \"redis.call('EXPIREAT', KEYS&#91;1], ARGV&#91;#ARGV])\";\n\n                    List&lt;String&gt; keys = Collections.singletonList(couponTemplateCacheKey);\n                    List&lt;String&gt; args = new ArrayList&lt;&gt;(actualCacheTargetMap.size() * 2 + 1);\n                    actualCacheTargetMap.forEach((key, value) -&gt; {\n                        args.add(key);\n                        args.add(value);\n                    });\n\n                    \/\/ \u4f18\u60e0\u5238\u6d3b\u52a8\u8fc7\u671f\u65f6\u95f4\u8f6c\u6362\u4e3a\u79d2\u7ea7\u522b\u7684 Unix \u65f6\u95f4\u6233\n                    args.add(String.valueOf(couponTemplateDO.getValidEndTime().getTime() \/ 1000));\n\n                    \/\/ \u6267\u884c LUA \u811a\u672c\n                    stringRedisTemplate.execute(\n                            new DefaultRedisScript&lt;&gt;(luaScript, Long.class),\n                            keys,\n                            args.toArray()\n                    );\n                    couponTemplateCacheMap = cacheTargetMap.entrySet()\n                            .stream()\n                            .collect(Collectors.toMap(Map.Entry::getKey, Map.Entry::getValue));\n                }\n            } finally {\n                lock.unlock();\n            }\n        }\n\n        return BeanUtil.mapToBean(couponTemplateCacheMap, CouponTemplateQueryRespDTO.class, false, CopyOptions.create());\n    }\n\n    \/**\n     * \u7f13\u5b58\u7a7f\u900f\u89e3\u51b3\u65b9\u6848\u4e4b\u5e03\u9686\u8fc7\u6ee4\u5668\n     *\/\n    public CouponTemplateQueryRespDTO findCouponTemplateV2(CouponTemplateQueryReqDTO requestParam) {\n        \/\/ \u5224\u65ad\u5e03\u9686\u8fc7\u6ee4\u5668\u662f\u5426\u5b58\u5728\u6307\u5b9a\u6a21\u677f ID\uff0c\u4e0d\u5b58\u5728\u76f4\u63a5\u8fd4\u56de\u9519\u8bef\n        if (!couponTemplateQueryBloomFilter.contains(requestParam.getCouponTemplateId())) {\n            throw new ClientException(\"\u4f18\u60e0\u5238\u6a21\u677f\u4e0d\u5b58\u5728\");\n        }\n\n        LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                .eq(CouponTemplateDO::getShopNumber, Long.parseLong(requestParam.getShopNumber()))\n                .eq(CouponTemplateDO::getId, Long.parseLong(requestParam.getCouponTemplateId()))\n                .eq(CouponTemplateDO::getStatus, CouponTemplateStatusEnum.ACTIVE.getStatus());\n        CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n\n        return BeanUtil.toBean(couponTemplateDO, CouponTemplateQueryRespDTO.class);\n    }\n}\n<\/code><\/pre>\n\n\n\n<p>\u5982\u679c\u6211\u4eec\u6309\u7167\u6b63\u5e38\u903b\u8f91\uff0c\u662f\u6ca1\u6709\u529e\u6cd5\u89e6\u53d1\u7f13\u5b58\u7a7a\u503c\u7b49\u64cd\u4f5c\u7684\u3002\u6240\u4ee5\u9700\u8981\u5728\u6b63\u5e38\u903b\u8f91\u4e0b\uff0c\u52a0\u4e00\u4e9b\u4eba\u5de5\u7834\u574f\uff0c\u6d41\u7a0b\u5982\u4e0b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u786e\u4fdd\u5546\u5bb6\u540e\u7ba1\u7cfb\u7edf\u5df2\u7ecf\u63a5\u5165\u4e86\u5e03\u9686\u8fc7\u6ee4\u5668\u6d41\u7a0b\uff0c\u901a\u8fc7\u5546\u5bb6\u540e\u7ba1\u7cfb\u7edf\u6dfb\u52a0\u65b0\u7684\u4f18\u60e0\u5238\uff1b<\/li>\n\n\n\n<li>2. \u53bb\u6570\u636e\u5e93\u628a\u521a\u624d\u90a3\u4e00\u6761\u8bb0\u5f55\u5220\u9664\u6216\u8005 status \u8bbe\u7f6e\u4e3a 1 \u4e5f\u5c31\u662f\u5df2\u7ed3\u675f\u72b6\u6001\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u901a\u8fc7\u8fd9\u79cd\u65b9\u5f0f\uff0c\u5927\u5bb6\u5c31\u80fd\u89e6\u53d1\u7c7b\u4f3c\u4e8e\u5e03\u9686\u8fc7\u6ee4\u5668\u8bef\u5224\uff0c\u7136\u540e\u89e6\u53d1\u7f13\u5b58\u7a7a\u503c\u548c\u5206\u5e03\u5f0f\u9501\u7b49\u6d41\u7a0b\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u5e38\u89c1\u95ee\u9898\u7b54\u7591<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u5e03\u9686\u8fc7\u6ee4\u5668\u8bbe\u7f6e\u591a\u5927\u5bb9\u91cf\uff1f<\/h3>\n\n\n\n<p>\u5e03\u9686\u8fc7\u6ee4\u5668\u7684\u5bb9\u91cf\u5c31\u53d6\u51b3\u4e8e\u4e1a\u52a1\u7684\u6570\u91cf\uff0c\u6211\u4eec\u4e4b\u524d\u5728\u5206\u5e93\u5206\u8868\u7684\u7ae0\u8282\u4e0a\u8bf4\uff0c\u53ef\u80fd\u6709 300 \u4ebf\u7684\u4f18\u60e0\u5238\u6a21\u677f\u6570\u91cf\uff0c\u662f\u4e0d\u662f\u76f4\u63a5\u8bbe\u7f6e\u4e3a\u5e03\u9686\u8fc7\u6ee4\u5668\u7684\u5bb9\u91cf\u5c31\u597d\u4e86\uff1f<\/p>\n\n\n\n<p>\u4e0d\u884c\u7684\uff0c\u4f60\u53ef\u4ee5\u5c1d\u8bd5\u4e0b\u8bbe\u7f6e 300 \u4ebf\u9884\u4f30\u5143\u7d20\u4ee5\u53ca\u5343\u5206\u4e4b\u4e00\u7684\u8bef\u5224\u7387\uff0c\u7edd\u5bf9\u4f1a\u62a5\u9519\u3002\u56e0\u4e3a<\/p>\n\n\n\n<p><code>Bloom filter size can't be greater than 4294967294. But calculated size is 431327626981<\/code><\/p>\n\n\n\n<p>\u7b80\u5355\u68b3\u7406\u4e86\u4e0b\uff0c\u610f\u601d\u662f\uff1a\u5e03\u9686\u8fc7\u6ee4\u5668\u5927\u5c0f\u4e0d\u80fd\u8d85\u8fc7 4294967294\uff0c\u4f46\u662f\u54b1\u4eec\u7684\u53c2\u6570 300 \u4ebf\u9884\u4f30\u503c\u548c \u5343\u5206\u4e4b\u4e00\u7684\u8bef\u5224\u7387\uff0c\u5df2\u7ecf\u8d85\u8fc7\u4e86\u8fd9\u4e2a\u6570\u636e\u3002<\/p>\n\n\n\n<p>\u5982\u679c\u8bbe\u7f6e 300 \u4ebf\u6570\u636e\u9884\u4f30\u503c\u4f46\u662f\u8bbe\u7f6e\u767e\u5206\u4e4b\u4e00\u7684\u8bef\u5224\u7387\uff0c\u90a3\u4e48\u62a5\u9519\u5c31\u6362\u4e86\u4e00\u4e2a\u3002\u53ef\u4ee5\u770b\u5230\u5e03\u9686\u8fc7\u6ee4\u5668\u7ed9\u6211\u4eec\u8bbe\u7f6e\u4e86\u4e0a\u9650\uff0c\u4e0d\u80fd\u8d85\u8fc7\u3002<\/p>\n\n\n\n<p><code>Bloom filter size can't be greater than 4294967294. But calculated size is 287551751321<\/code><\/p>\n\n\n\n<p>\u5c0f\u77e5\u8bc6\u70b9\uff0c\u4e00\u4e2a\u4ebf\u7684\u5143\u7d20\uff0c\u5982\u679c\u5343\u5206\u4e4b\u4e00\u7684\u8bef\u5224\u7387\uff0c\u90a3\u4e48\u5b9e\u9645\u5bb9\u91cf\u5927\u6982\u5728 170M \u5de6\u53f3\u3002\u53e6\u5916\u5728\u5bf9\u5e03\u9686\u8fc7\u6ee4\u5668\u8fdb\u884c\u521d\u59cb\u5316\u7684\u65f6\u5019\uff0c\u4f1a\u4e00\u6b21\u6027\u7533\u8bf7\u5bf9\u5e94\u7684\u5185\u5b58\uff0c\u8fd9\u4e2a\u9700\u8981\u989d\u5916\u6ce8\u610f\u4e0b\uff0c\u907f\u514d\u521d\u59cb\u5316\u8d85\u5927\u5bb9\u91cf\u5e03\u9686\u8fc7\u6ee4\u5668\u65f6\u5185\u5b58\u4e0d\u8db3\u95ee\u9898\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"896\" width=\"2132\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/954e1480ea8b110721b975e9ed3c22b8.png\" alt=\"\"><\/p>\n\n\n\n<p>\u90a3\u8fd9\u79cd\u60c5\u51b5\u4e0b\u5982\u4f55\u89e3\u51b3\u5e03\u9686\u8fc7\u6ee4\u5668\u4e0d\u80fd\u8bbe\u7f6e 300 \u4ebf\u6570\u636e\u95ee\u9898\uff1f<\/p>\n\n\n\n<p>\u53ef\u4ee5\u50cf\u4e4b\u524d\u5728\u5904\u7406\u7f13\u5b58\u51fb\u7a7f\u65f6\u6240\u7528\u7684\u5206\u7247\u903b\u8f91\u4e00\u6837\uff0c\u8bbe\u7f6e\u591a\u4e2a\u5e03\u9686\u8fc7\u6ee4\u5668\uff0c\u4f7f\u5f97\u8fd9\u4e9b\u5206\u7247\u7684\u5e03\u9686\u8fc7\u6ee4\u5668\u603b\u5bb9\u91cf\u80fd\u8fbe\u5230 300 \u4ebf\u3002\u7136\u540e\u6839\u636e\u6a21\u677f ID \u8fdb\u884c\u5206\u7247\uff0c\u786e\u5b9a\u8981\u64cd\u4f5c\u7684\u5e03\u9686\u8fc7\u6ee4\u5668\uff0c\u4ece\u800c\u5728\u8be5\u5206\u7247\u4e0a\u8fdb\u884c\u64cd\u4f5c\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-lock\">2. \u5206\u5e03\u5f0f\u9501 lock \u4f1a\u89e6\u53d1\u957f\u65f6\u95f4\u963b\u585e\u4e48\uff1f<\/h3>\n\n\n\n<p>\u8fd9\u4e2a\u903b\u8f91\u548c\u7f13\u5b58\u51fb\u7a7f\u91cc\u7684\u903b\u8f91\u662f\u4e00\u6837\u7684\uff0c\u5927\u5bb6\u53ef\u4ee5\u53c2\u8003\u7f13\u5b58\u51fb\u7a7f\u7ae0\u8282\u4e2d\u7684 tryLock \u548c\u5206\u5e03\u5f0f\u9501\u5206\u7247\u5904\u7406\u65b9\u6848\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u7b2c18\u5c0f\u8282\uff1a\u5982\u4f55\u8bbe\u7f6eRedis\u5185\u5b58\u6dd8\u6c70\u7b56\u7565\uff1f<\/h2>\n\n\n\n<h2 class=\"wp-block-heading\">\u4e1a\u52a1\u80cc\u666f<\/h2>\n\n\n\n<p>\u5728\u4e4b\u524d\u8ba8\u8bba\u7f13\u5b58\u51fb\u7a7f\u7684\u95ee\u9898\u65f6\uff0c\u6709\u4e00\u4e2a\u5173\u952e\u70b9\u6ca1\u6709\u8be6\u7ec6\u4ecb\u7ecd\uff0c\u90a3\u5c31\u662f\u5982\u4f55\u4e0e Redis \u7684\u5185\u5b58\u6dd8\u6c70\u7b56\u7565\u7ed3\u5408\u4f7f\u7528\u3002\u4ec5\u4ec5\u4f9d\u9760\u7f13\u5b58\u9884\u70ed\u548c\u8bbe\u7f6e\u6570\u636e\u6c38\u4e0d\u8fc7\u671f\u5e76\u4e0d\u662f\u4e00\u4e2a\u5168\u9762\u7684\u89e3\u51b3\u65b9\u6848\uff0c\u8fd8\u9700\u8981\u8003\u8651\u5408\u9002\u7684\u5185\u5b58\u6dd8\u6c70\u7b56\u7565\u6765\u4fdd\u969c\u7f13\u5b58\u51fb\u7a7f\u65b9\u6848\u6709\u6548\u6267\u884c\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4ec0\u4e48\u662f\u5185\u5b58\u6dd8\u6c70\uff1f<\/h2>\n\n\n\n<p>\u5185\u5b58\u603b\u662f\u6709\u9650\u7684\uff0c\u56e0\u6b64\u5f53 Redis \u5185\u5b58\u8d85\u51fa\u6700\u5927\u5185\u5b58\u65f6\uff0c\u5c31\u9700\u8981\u6839\u636e\u4e00\u5b9a\u7684\u7b56\u7565\u53bb\u4e3b\u52a8\u7684\u6dd8\u6c70\u4e00\u4e9b Key\uff0c\u6765\u817e\u51fa\u5185\u5b58\uff0c\u8fd9\u5c31\u662f\u5185\u5b58\u6dd8\u6c70\u7b56\u7565\u3002\u6211\u4eec\u53ef\u4ee5\u5728\u914d\u7f6e\u6587\u4ef6\u4e2d\u901a\u8fc7 <code>maxmemory-policy<\/code> \u914d\u7f6e\u6307\u5b9a\u7b56\u7565\u3002<\/p>\n\n\n\n<p>\u4e0e\u5230\u671f\u5220\u9664\u7b56\u7565\u4e0d\u540c\uff0c\u5185\u5b58\u6dd8\u6c70\u7b56\u7565\u4e3b\u8981\u76ee\u7684\u5219\u662f\u4e3a\u4e86\u9632\u6b62\u8fd0\u884c\u65f6\u5185\u5b58\u8d85\u8fc7\u6700\u5927\u5185\u5b58\uff0c\u6240\u4ee5\u5c3d\u7ba1\u6700\u7ec8\u76ee\u7684\u90fd\u662f\u6e05\u7406\u5185\u5b58\u4e2d\u7684\u4e00\u4e9b Key\uff0c\u4f46\u662f\u5b83\u4eec\u7684\u5e94\u7528\u573a\u666f\u548c\u89e6\u53d1\u65f6\u673a\u662f\u4e0d\u540c\u7684\u3002<\/p>\n\n\n\n<p>\u7b97\u4e0a\u5728 4.0 \u6dfb\u52a0\u7684\u4e24\u79cd\u57fa\u4e8e LFU \u7b97\u6cd5\u7684\u7b56\u7565\uff0c Redis \u4e00\u5171\u63d0\u4f9b\u4e86\u516b\u79cd\u7b56\u7565\u4f9b\u6211\u4eec\u9009\u62e9\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li><code>noeviction<\/code>\uff0c\u4e0d\u6dd8\u6c70\u4efb\u4f55 key\uff0c\u76f4\u63a5\u62a5\u9519\u3002\u5b83\u662f\u9ed8\u8ba4\u7b56\u7565\u3002<\/li>\n\n\n\n<li><code>volatile-random<\/code>\uff1a\u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u968f\u673a\u6dd8\u6c70\u4e00\u4e2a Key\u3002<\/li>\n\n\n\n<li><code>volatile-lru<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\u7684 Key\u3002<\/li>\n\n\n\n<li><code>volatile-lfu<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u4e0d\u5e38\u7528\u4f7f\u7528\u7684 Key\uff084.0 \u65b0\u589e\uff09\u3002<\/li>\n\n\n\n<li><code>volatile-ttl<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u4f18\u5148\u6dd8\u6c70\u6700\u65e9\u8fc7\u671f\u7684 Key\u3002<\/li>\n\n\n\n<li><code>allkeys-random<\/code>\uff1a\u4ece\u6240\u6709 Key \u4e2d\uff0c\u968f\u673a\u6dd8\u6c70\u4e00\u4e2a\u952e\u3002<\/li>\n\n\n\n<li><code>allkeys-lru<\/code>\uff1a \u4ece\u6240\u6709 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\u7684 Key\u3002<\/li>\n\n\n\n<li><code>allkeys-lfu<\/code>\uff1a \u4ece\u6240\u6709 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u4e0d\u5e38\u7528\u4f7f\u7528\u7684\u952e\uff084.0 \u65b0\u589e\uff09\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u4ece\u6dd8\u6c70\u8303\u56f4\u6765\u8bf4\u53ef\u4ee5\u5206\u4e3a\u4e0d\u6dd8\u6c70\u4efb\u4f55\u6570\u636e\u3001\u53ea\u4ece\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684\u952e\u4e2d\u6dd8\u6c70\u548c\u4ece\u6240\u6709\u952e\u4e2d\u6dd8\u6c70\u4e09\u7c7b\u3002\u800c\u4ece\u6dd8\u6c70\u7b97\u6cd5\u6765\u5206\uff0c\u53c8\u4e3b\u8981\u5206\u4e3a Random\uff08\u968f\u673a\uff09\uff0cLRU\uff08\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\uff09\uff0c\u4ee5\u53ca LFU\uff08\u6700\u8fd1\u6700\u4e0d\u5e38\u4f7f\u7528\uff09\u4e09\u79cd\u3002<\/p>\n\n\n\n<p>\u5176\u4e2d\uff0c\u5173\u4e8e LRU \u7b97\u6cd5\uff0c\u5b83\u662f\u4e00\u79cd\u975e\u5e38\u5e38\u89c1\u7684\u7f13\u5b58\u6dd8\u6c70\u7b97\u6cd5\u3002\u6211\u4eec\u53ef\u4ee5\u7b80\u5355\u7406\u89e3\u4e3a Redis \u4f1a\u5728\u6bcf\u6b21\u8bbf\u95ee Key \u7684\u65f6\u5019\u8bb0\u5f55\u8bbf\u95ee\u65f6\u95f4\uff0c\u5f53\u6dd8\u6c70\u65f6\uff0c\u4f18\u5148\u6dd8\u6c70\u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u8ddd\u79bb\u73b0\u5728\u6700\u65e9\u7684 Key\u3002<\/p>\n\n\n\n<p>\u800c\u5bf9\u4e8e LFU \u7b97\u6cd5\uff0c\u6211\u4eec\u53ef\u4ee5\u7406\u89e3\u4e3a Redis \u4f1a\u5728\u8bbf\u95ee Key \u65f6\uff0c\u6839\u636e\u4e24\u6b21\u8bbf\u95ee\u65f6\u95f4\u7684\u95f4\u9694\u8ba1\u7b97\u5e76\u7d2f\u52a0\u8bbf\u95ee\u9891\u7387\u6307\u6807\uff0c\u5f53\u6dd8\u6c70\u65f6\uff0c\u4f18\u5148\u6dd8\u6c70\u8bbf\u95ee\u9891\u7387\u6307\u6807\u6700\u4f4e\u7684 key\u3002\u76f8\u6bd4 LRU \u7b97\u6cd5\uff0c\u5b83\u907f\u514d\u4e86\u4f4e\u9891\u7387\u7684\u5927\u6279\u91cf\u67e5\u8be2\u9020\u6210\u7684\u7f13\u5b58\u6c61\u67d3\u95ee\u9898\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4ec0\u4e48\u662f\u7f13\u5b58\u6c61\u67d3\u95ee\u9898\uff1f<\/h2>\n\n\n\n<p>LRU \u6709\u4e2a\u6700\u5927\u95ee\u9898\uff0c\u5c31\u662f\u5b83\u53ea\u8ba4\u6700\u8fd1\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u3002\u800c\u5982\u679c\u51fa\u73b0\u7cfb\u7edf\u5076\u5c14\u9700\u8981\u4e00\u6b21\u6027\u8bfb\u53d6\u5927\u91cf\u6570\u636e\u7684\u65f6\u5019\uff0c\u4f1a\u5927\u89c4\u6a21\u66f4\u65b0 Key \u7684\u6700\u8fd1\u8bbf\u95ee\u65f6\u95f4\uff0c\u4ece\u800c\u5bfc\u81f4\u771f\u6b63\u9700\u8981\u88ab\u9891\u7e41\u8bbf\u95ee\u7684 Key \u56e0\u4e3a\u6700\u8fd1\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u66f4\u65e9\u800c\u88ab\u76f4\u63a5\u6dd8\u6c70\u3002\u8fd9\u79cd\u60c5\u51b5\u88ab\u79f0\u4e3a\u7f13\u5b58\u6c61\u67d3\u3002\u4e3a\u6b64\uff0c\u6211\u4eec\u9700\u8981\u4f7f\u7528 LFU \u7b97\u6cd5\u6765\u89e3\u51b3\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6dd8\u6c70\u7b56\u7565\u5bf9\u7f13\u5b58\u51fb\u7a7f\u7684\u5f71\u54cd\uff1f<\/h2>\n\n\n\n<p>\u4e3e\u4e2a\u4f8b\u5b50\uff0c\u6211\u4eec\u64cd\u4f5c\u4e86\u7f13\u5b58\u9884\u70ed\u548c\u8bbe\u7f6e Redis \u6c38\u4e0d\u8fc7\u671f\uff0c\u5982\u679c\u8bf4\u8bbe\u7f6e Redis \u5185\u5b58\u6dd8\u6c70\u7b56\u7565\u662f\u4ee5\u4e0b\u7c7b\u578b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li><code>allkeys-random<\/code>\uff1a\u4ece\u6240\u6709 Key \u4e2d\uff0c\u968f\u673a\u6dd8\u6c70\u4e00\u4e2a\u952e\u3002<\/li>\n\n\n\n<li><code>allkeys-lru<\/code>\uff1a \u4ece\u6240\u6709 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\u7684 Key\u3002<\/li>\n\n\n\n<li><code>allkeys-lfu<\/code>\uff1a \u4ece\u6240\u6709 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u4e0d\u5e38\u7528\u4f7f\u7528\u7684\u952e\uff084.0 \u65b0\u589e\uff09\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u56e0\u4e3a\u662f\u4ece\u6240\u6709\u952e\u4e2d\u53bb\u6267\u884c\u6dd8\u6c70\u7b97\u6cd5\uff0c\u662f\u5426\u6709\u53ef\u80fd\u5c06\u54b1\u4eec\u8bbe\u7f6e\u7684\u70ed Key \u6216\u8005\u8bf4\u8bbf\u95ee\u8f83\u591a\u7684 Key \u7ed9\u6dd8\u6c70\u6389\uff1f\u7b2c\u4e00\u79cd\u7edd\u5bf9\u6709\u53ef\u80fd\uff0c\u7b2c\u4e8c\u4e09\u79cd\u867d\u7136\u6982\u7387\u6bd4\u8f83\u5c0f\uff0c\u4f46\u662f\u4e5f\u4e0d\u662f\u6ca1\u53ef\u80fd\u3002<\/p>\n\n\n\n<p>\u6240\u4ee5\uff0c\u6211\u4eec\u53ea\u80fd\u4ece\u8fd9\u51e0\u4e2a\u6dd8\u6c70\u7b97\u6cd5\u4e2d\u8fdb\u884c\u9009\u62e9\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li><code>noeviction<\/code>\uff0c\u4e0d\u6dd8\u6c70\u4efb\u4f55 Key\uff0c\u76f4\u63a5\u62a5\u9519\u3002<\/li>\n\n\n\n<li><code>volatile-random<\/code>\uff1a\u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u968f\u673a\u6dd8\u6c70\u4e00\u4e2a Key\u3002<\/li>\n\n\n\n<li><code>volatile-lru<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\u7684 Key\u3002<\/li>\n\n\n\n<li><code>volatile-lfu<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u6dd8\u6c70\u6700\u8fd1\u6700\u4e0d\u5e38\u7528\u4f7f\u7528\u7684 Key\uff084.0 \u65b0\u589e\uff09\u3002<\/li>\n\n\n\n<li><code>volatile-ttl<\/code>\uff1a \u4ece\u6240\u6709\u8bbe\u7f6e\u4e86\u5230\u671f\u65f6\u95f4\u7684 Key \u4e2d\uff0c\u4f18\u5148\u6dd8\u6c70\u6700\u65e9\u8fc7\u671f\u7684 Key\u3002<\/li>\n<\/ul>\n\n\n\n<h2 class=\"wp-block-heading\">\u725b\u5238\u9009\u62e9\u54ea\u4e2a\u6dd8\u6c70\u7b56\u7565\uff1f<\/h2>\n\n\n\n<p>\u4ece\u5e38\u89c4\u4e0a\u6765\u8bf4\uff0c\u5982\u679c\u5927\u5bb6\u516c\u53f8 Redis \u6bd4\u8f83\u8c6a\uff0c\u4e5f\u5c31\u662f\u4e1a\u52a1\u6323\u94b1\uff0c\u5728\u8d44\u6e90\u4e0a\u4e0d\u541d\u556c\uff0c\u5fc5\u7136\u6beb\u4e0d\u72b9\u8c6b\u9009\u62e9 <code>noeviction<\/code> \u4e0d\u6dd8\u6c70\u4efb\u4f55\u4efb\u4f55\u6570\u636e\u3002<\/p>\n\n\n\n<p>\u7b80\u5355\u7684\u6765\u8bf4\uff0c\u5982\u679c\u4f60\u7684\u4e1a\u52a1\u6570\u636e\u7684\u8bbf\u95ee\u6bd4\u8f83\u5e73\u5747\uff0c\u4e0d\u5b58\u5728\u660e\u663e\u7684\u51b7\u70ed\u533a\u522b\uff0c\u90a3\u4e48 LRU \u53ef\u4ee5\u6ee1\u8db3\u4e00\u822c\u7684\u4f7f\u7528\u9700\u6c42\u3002\u5982\u679c\u4f60\u7684\u4e1a\u52a1\u5177\u5907\u5f88\u5f3a\u7684\u65f6\u6548\u6027\uff0c\u800c\u4e14\u662f\u5b58\u5728\u5927\u4fc3\u5546\u54c1\u8fd9\u79cd\u660e\u663e\u7684\u70ed\u70b9\u6570\u636e\uff0c\u90a3\u4e48\u63a8\u8350\u4f60\u4f7f\u7528 LFU\u3002<\/p>\n\n\n\n<p>\u56e0\u4e3a\u725b\u5238\u91cc\u7684\u4e00\u4e9b\u6a21\u677f\u4e5f\u4e0d\u5168\u662f\u5e38\u7528\u7684\uff0c\u9002\u5f53\u5185\u5b58\u5403\u7d27\u7684\u65f6\u5019\u53ef\u4ee5\u6dd8\u6c70\u4e00\u90e8\u5206\u5e26\u8fc7\u671f\u65f6\u95f4\u7684 Key\uff0c\u518d\u52a0\u4e0a\u725b\u5238\u4e2d\u7684\u4f18\u60e0\u5238\u5b58\u5728\u660e\u663e\u7684\u51b7\u70ed\u6570\u636e\u73b0\u8c61\uff0c\u6211\u5efa\u8bae\u4f7f\u7528 <code>volatile-lfu<\/code>\uff0c\u53ef\u4ee5\u6709\u6548\u907f\u514d LRU \u7684\u7f13\u5b58\u6c61\u67d3\u95ee\u9898\u3002<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u4ee5\u4e0b\u5185\u5bb9\u4e86\u89e3\u5373\u53ef\uff0c\u5982\u4e0d\u5bb9\u6613\u7406\u89e3\uff0c\u53ef\u8df3\u8fc7\u77e5\u9053\u5982\u4f55\u5e94\u7528\u5373\u53ef\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u5185\u5b58\u6dd8\u6c70\u5e95\u5c42\u539f\u7406<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u6dd8\u6c70\u8fc7\u7a0b<\/h3>\n\n\n\n<p>Redis \u5185\u5b58\u6dd8\u6c70\u6267\u884c\u6d41\u7a0b\u5982\u4e0b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u6bcf\u6b21\u5f53 Redis \u6267\u884c\u547d\u4ee4\u65f6\uff0c\u82e5\u8bbe\u7f6e\u4e86\u6700\u5927\u5185\u5b58\u5927\u5c0f <code>maxmemory<\/code>\uff0c\u5e76\u8bbe\u7f6e\u4e86\u6dd8\u6c70\u7b56\u7565\u5f0f\uff0c\u5219\u4f1a\u5c1d\u8bd5\u8fdb\u884c\u4e00\u6b21 Key \u6dd8\u6c70\uff1b<\/li>\n\n\n\n<li>2. Redis \u9996\u5148\u4f1a\u8bc4\u4f30\u5df2\u4f7f\u7528\u5185\u5b58\uff08\u8fd9\u91cc\u4e0d\u5305\u542b\u4e3b\u4ece\u590d\u5236\u4f7f\u7528\u7684\u4e24\u4e2a\u7f13\u51b2\u533a\u5360\u7528\u7684\u5185\u5b58\uff09\u662f\u5426\u5927\u4e8e <code>maxmemory<\/code>\uff0c\u5982\u679c\u6ca1\u6709\u5219\u76f4\u63a5\u8fd4\u56de\uff0c\u5426\u5219\u5c06\u8ba1\u7b97\u5f53\u524d\u9700\u8981\u91ca\u653e\u591a\u5c11\u5185\u5b58\uff0c\u968f\u540e\u5f00\u59cb\u6839\u636e\u7b56\u7565\u6dd8\u6c70\u7b26\u5408\u6761\u4ef6\u7684 Key\uff1b\u5f53\u5f00\u59cb\u8fdb\u884c\u6dd8\u6c70\u65f6\uff0c\u5c06\u4f1a\u4f9d\u6b21\u5bf9\u6bcf\u4e2a\u6570\u636e\u5e93\u8fdb\u884c\u62bd\u6837\uff0c\u62bd\u6837\u7684\u6570\u636e\u8303\u56f4\u7531\u7b56\u7565\u51b3\u5b9a\uff0c\u800c\u6837\u672c\u6570\u91cf\u5219\u7531 <code>maxmemory-samples<\/code>\u914d\u7f6e\u51b3\u5b9a\uff1b<\/li>\n\n\n\n<li>3. \u5b8c\u6210\u62bd\u6837\u540e\uff0cRedis \u4f1a\u5c1d\u8bd5\u5c06\u6837\u672c\u653e\u5165\u63d0\u524d\u521d\u59cb\u5316\u597d <code>EvictionPoolLRU<\/code> \u6570\u7ec4\u4e2d\uff0c\u5b83\u76f8\u5f53\u4e8e\u4e00\u4e2a\u4e34\u65f6\u7f13\u51b2\u533a\uff0c\u5f53\u6570\u7ec4\u586b\u6ee1\u4ee5\u540e\u5373\u5c06\u91cc\u9762\u5168\u90e8\u7684 Key \u8fdb\u884c\u5220\u9664\u3002<\/li>\n\n\n\n<li>4. \u82e5\u4e00\u6b21\u5220\u9664\u540e\u5185\u5b58\u4ecd\u7136\u4e0d\u8db3\uff0c\u5219\u518d\u6b21\u91cd\u590d\u4e0a\u4e00\u6b65\u9aa4\uff0c\u5c06\u6837\u672c\u4e2d\u7684\u5269\u4f59 Key \u518d\u6b21\u586b\u5165\u6570\u7ec4\u4e2d\u8fdb\u884c\u5220\u9664\uff0c\u76f4\u5230\u91ca\u653e\u4e86\u8db3\u591f\u7684\u5185\u5b58\uff0c\u6216\u8005\u672c\u6b21\u62bd\u6837\u7684\u6240\u6709 Key \u90fd\u88ab\u5220\u9664\u5b8c\u6bd5\uff08\u5982\u679c\u6b64\u65f6\u5185\u5b58\u8fd8\u662f\u4e0d\u8db3\uff0c\u90a3\u4e48\u5c31\u91cd\u65b0\u6267\u884c\u4e00\u6b21\u6dd8\u6c70\u6d41\u7a0b\uff09\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5728\u62bd\u6837\u8fd9\u4e00\u6b65\uff0c\u6d89\u53ca\u5230\u4ece\u5b57\u5178\u4e2d\u968f\u673a\u62bd\u6837\u8fd9\u4e2a\u8fc7\u7a0b\uff0c\u7531\u4e8e\u54c8\u5e0c\u8868\u7684 Key \u662f\u6563\u5217\u5206\u5e03\u7684\uff0c\u56e0\u6b64\u4f1a\u6709\u5f88\u591a\u6876\u90fd\u662f\u7a7a\u7684\uff0c\u7eaf\u968f\u673a\u6548\u7387\u53ef\u80fd\u4f1a\u5f88\u4f4e\u3002\u56e0\u6b64\uff0cRedis \u91c7\u7528\u4e86\u4e00\u4e2a\u7279\u522b\u7684\u505a\u6cd5\uff0c\u90a3\u5c31\u662f\u5148\u8fde\u7eed\u904d\u5386\u6570\u4e2a\u6876\uff0c\u5982\u679c\u90fd\u662f\u7a7a\u7684\uff0c\u518d\u968f\u673a\u8c03\u5230\u53e6\u4e00\u4e2a\u4f4d\u7f6e\uff0c\u518d\u8fde\u7eed\u904d\u5386\u51e0\u4e2a\u6876\u2026\u2026\u5982\u6b64\u5faa\u73af\uff0c\u76f4\u5230\u7ed3\u675f\u62bd\u6837\u3002<\/p>\n\n\n\n<p>\u4f60\u53ef\u4ee5\u53c2\u7167\u6e90\u7801\u7406\u89e3\u8fd9\u4e2a\u8fc7\u7a0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>unsigned int dictGetSomeKeys(dict *d, dictEntry **des, unsigned int count) {\n    unsigned long j; \/* internal hash table id, 0 or 1. *\/\n    unsigned long tables; \/* 1 or 2 tables? *\/\n    unsigned long stored = 0, maxsizemask;\n    unsigned long maxsteps;\n\u200b\n    if (dictSize(d) &lt; count) count = dictSize(d);\n    maxsteps = count*10;\n\u200b\n    \/\/ \u5982\u679c\u5b57\u5178\u6b63\u5728\u8fc1\u79fb\uff0c\u5219\u534f\u52a9\u8fc1\u79fb\n    for (j = 0; j &lt; count; j++) {\n        if (dictIsRehashing(d))\n            _dictRehashStep(d);\n        else\n            break;\n    }\n\u200b\n    tables = dictIsRehashing(d) ? 2 : 1;\n    maxsizemask = d-&gt;ht&#91;0].sizemask;\n    if (tables &gt; 1 &amp;&amp; maxsizemask &lt; d-&gt;ht&#91;1].sizemask)\n        maxsizemask = d-&gt;ht&#91;1].sizemask;\n\u200b\n    unsigned long i = random() &amp; maxsizemask;\n    unsigned long emptylen = 0;\n\u200b\n    \/\/ \u5f53\u5df2\u7ecf\u91c7\u96c6\u5230\u8db3\u591f\u7684\u6837\u672c\uff0c\u6216\u8005\u91cd\u8bd5\u5df2\u8fbe\u4e0a\u9650\u5219\u7ed3\u675f\u91c7\u6837\n    while(stored &lt; count &amp;&amp; maxsteps--) {\n        for (j = 0; j &lt; tables; j++) {\n            if (tables == 2 &amp;&amp; j == 0 &amp;&amp; i &lt; (unsigned long) d-&gt;rehashidx) {\n                if (i &gt;= d-&gt;ht&#91;1].size)\n                    i = d-&gt;rehashidx;\n                else\n                    continue;\n            }\n\u200b\n            \/\/ \u5982\u679c\u4e00\u4e2a\u5e93\u7684\u5230\u671f\u5b57\u5178\u5df2\u7ecf\u5904\u7406\u5b8c\u6bd5\uff0c\u5219\u5904\u7406\u4e0b\u4e00\u4e2a\u5e93\n            if (i &gt;= d-&gt;ht&#91;j].size) continue;\n            dictEntry *he = d-&gt;ht&#91;j].table&#91;i];\n\u200b\n            \/\/ \u8fde\u7eed\u904d\u5386\u591a\u4e2a\u6876\uff0c\u5982\u679c\u591a\u4e2a\u6876\u90fd\u662f\u7a7a\u7684\uff0c\u90a3\u4e48\u968f\u673a\u8df3\u5230\u53e6\u4e00\u4e2a\u4f4d\u7f6e\uff0c\u7136\u540e\u518d\u91cd\u590d\u6b64\u6b65\u9aa4           \n            if (he == NULL) {\n                emptylen++;\n                if (emptylen &gt;= 5 &amp;&amp; emptylen &gt; count) {\n                    i = random() &amp; maxsizemask;\n                    emptylen = 0;\n                }\n            } else {\n                emptylen = 0;\n                while (he) {\n                    *des = he;\n                    des++;\n                    he = he-&gt;next;\n                    stored++;\n                    if (stored == count) return stored;\n                }\n            }\n        }\n\u200b\n        \/\/ \u67e5\u627e\u4e0b\u4e00\u4e2a\u6876\n        i = (i+1) &amp; maxsizemask;\n    }\n    return stored;\n}<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-lru\">2. LRU \u5b9e\u73b0<\/h3>\n\n\n\n<p>LRU \u7684\u5168\u79f0\u4e3a <code>Least Recently Used<\/code>\uff0c\u4e5f\u5c31\u662f\u6700\u8fd1\u6700\u5c11\u4f7f\u7528\u3002\u4e00\u822c\u6765\u8bf4\uff0cLRU \u4f1a\u4ece\u4e00\u6279 Key \u4e2d\u6dd8\u6c70\u4e0a\u6b21\u8bbf\u95ee\u65f6\u95f4\u6700\u65e9\u7684 key\u3002<\/p>\n\n\n\n<p>\u5b83\u662f\u4e00\u79cd\u975e\u5e38\u5e38\u89c1\u7684\u7f13\u5b58\u56de\u6536\u7b97\u6cd5\uff0c\u5728\u8bf8\u5982 <code>Guava Cache<\/code>\u3001<code>Caffeine<\/code>\u7b49\u7f13\u5b58\u5e93\u4e2d\u90fd\u63d0\u4f9b\u4e86\u7c7b\u4f3c\u7684\u5b9e\u73b0\u3002\u6211\u4eec\u81ea\u5df1\u4e5f\u53ef\u4ee5\u57fa\u4e8e JDK \u7684 <code>LinkedHashMap<\/code> \u5b9e\u73b0\u652f\u6301 LRU \u7b97\u6cd5\u7684\u7f13\u5b58\u529f\u80fd\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-1-lru\">2.1 \u8fd1\u4f3c LRU<\/h4>\n\n\n\n<p>\u4f20\u7edf\u7684 LRU \u7b97\u6cd5\u5b9e\u73b0\u901a\u5e38\u4f1a\u7ef4\u62a4\u4e00\u4e2a\u94fe\u8868\uff0c\u5f53\u8bbf\u95ee\u8fc7\u67d0\u4e2a\u8282\u70b9\u540e\u5c31\u5c06\u8be5\u8282\u70b9\u79fb\u81f3\u94fe\u8868\u5934\u90e8\u3002\u5982\u6b64\u53cd\u590d\u540e\uff0c\u94fe\u8868\u7684\u8282\u70b9\u5c31\u4f1a\u6309\u6700\u8fd1\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u6392\u5e8f\u3002\u5f53\u7f13\u5b58\u6570\u91cf\u5230\u8fbe\u4e0a\u9650\u540e\uff0c\u6211\u4eec\u76f4\u63a5\u79fb\u9664\u5c3e\u8282\u70b9\uff0c\u5373\u53ef\u79fb\u9664\u6700\u8fd1\u6700\u5c11\u8bbf\u95ee\u7684\u7f13\u5b58\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"397\" width=\"407\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/f9a85633ef9c457710f624cd90bda6a9.png\" alt=\"\">\u200b<\/p>\n\n\n\n<p>\u4e0d\u8fc7\uff0c\u5bf9\u4e8e Redis \u6765\u8bf4\uff0c\u5982\u679c\u6bcf\u4e2a Key \u6dfb\u52a0\u7684\u65f6\u5019\u90fd\u9700\u8981\u989d\u5916\u7684\u7ef4\u62a4\u5e76\u64cd\u4f5c\u8fd9\u6837\u4e00\u6761\u94fe\u8868\uff0c\u8981\u989d\u5916\u4ed8\u51fa\u7684\u4ee3\u4ef7\u663e\u7136\u662f\u4e0d\u53ef\u63a5\u53d7\u7684\uff0c\u56e0\u6b64 Redis \u4e2d\u7684 LRU \u662f\u8fd1\u4f3c LRU\uff08<code>NearlyLRU<\/code>\uff09\u3002<\/p>\n\n\n\n<p>\u5f53\u6bcf\u6b21\u8bbf\u95ee Key \u65f6\uff0cRedis \u4f1a\u5728\u7ed3\u6784\u4f53\u4e2d\u8bb0\u5f55\u672c\u6b21\u8bbf\u95ee\u65f6\u95f4\uff0c\u800c\u5f53\u9700\u8981\u6dd8\u6c70 Key \u65f6\uff0c\u5c06\u4f1a\u4ece\u5168\u90e8\u6570\u636e\u4e2d\u8fdb\u884c\u62bd\u6837\uff0c\u7136\u540e\u518d\u79fb\u9664\u6837\u672c\u4e2d\u4e0a\u6b21\u8bbf\u95ee\u65f6\u95f4\u6700\u65e9\u7684 key\u3002<\/p>\n\n\n\n<p>\u5b83\u7684\u7279\u70b9\u662f\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u4ec5\u5f53\u9700\u8981\u65f6\u518d\u62bd\u6837\uff0c\u56e0\u800c\u4e0d\u9700\u8981\u7ef4\u62a4\u5168\u91cf\u6570\u636e\u7ec4\u6210\u7684\u94fe\u8868\uff0c\u8fd9\u907f\u514d\u4e86\u989d\u5916\u5185\u5b58\u6d88\u8017\u3002<\/li>\n\n\n\n<li>\u8bbf\u95ee\u65f6\u4ec5\u5728\u7ed3\u6784\u4f53\u4e0a\u8bb0\u5f55\u64cd\u4f5c\u65f6\u95f4\uff0c\u800c\u4e0d\u9700\u8981\u64cd\u4f5c\u94fe\u8868\u8282\u70b9\uff0c\u8fd9\u907f\u514d\u4e86\u989d\u5916\u7684\u6027\u80fd\u6d88\u8017\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5f53\u7136\uff0c\u6709\u5229\u5c31\u6709\u5f0a\uff0c\u8fd9\u79cd\u5b9e\u73b0\u65b9\u5f0f\u4e5f\u51b3\u5b9a Redis \u7684 LRU \u662f\u5e76\u4e0d\u662f\u767e\u5206\u767e\u51c6\u786e\u7684\uff0c\u88ab\u6dd8\u6c70\u7684 Key \u672a\u5fc5\u771f\u7684\u5c31\u662f\u6240\u6709 Key \u4e2d\u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u6700\u65e9\u7684\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"463\" width=\"638\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/b7eb3c560f8ad528f39dadf913c2f8a8.png\" alt=\"\"><\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"2-2\">2.2 \u62bd\u6837\u5927\u5c0f<\/h4>\n\n\n\n<p>\u6839\u636e\u4e0a\u8ff0\u7684\u5185\u5bb9\uff0c\u6211\u4eec\u4e0d\u96be\u7406\u89e3\uff0c\u5f53\u62bd\u6837\u7684\u6570\u91cf\u8d8a\u5927\uff0cLRU \u6dd8\u6c70 Key \u5c31\u8d8a\u51c6\u786e\uff0c\u76f8\u5bf9\u7684\u5f00\u9500\u4e5f\u66f4\u5927\u3002\u56e0\u6b64\uff0cRedis \u5141\u8bb8\u6211\u4eec\u901a\u8fc7 <code>maxmemory-samples<\/code> \u914d\u7f6e\u91c7\u6837\u6570\u91cf\uff08\u9ed8\u8ba4\u4e3a 5\uff09\uff0c\u4ece\u800c\u5728\u6027\u80fd\u548c\u7cbe\u5ea6\u4e0a\u53d6\u5f97\u5e73\u8861\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3-lfu\">3. LFU \u5b9e\u73b0<\/h3>\n\n\n\n<p>LFU \u5168\u79f0\u4e3a <code>Least Frequently Used<\/code> \uff0c\u4e5f\u5c31\u662f\u6700\u8fd1\u6700\u4e0d\u5e38\u7528\u3002\u5b83\u7684\u7279\u70b9\u5982\u4e0b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u540c\u6837\u662f\u57fa\u4e8e\u62bd\u6837\u5b9e\u73b0\u7684\u8fd1\u4f3c\u7b97\u6cd5\uff0c<code>maxmemory-samples<\/code> \u5bf9\u5176\u540c\u6837\u6709\u6548\u3002<\/li>\n\n\n\n<li>\u6bd4\u8f83\u7684\u4e0d\u662f\u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\uff0c\u800c\u662f\u6570\u636e\u7684\u8bbf\u95ee\u9891\u7387\u3002\u5f53\u6dd8\u6c70\u7684\u65f6\u5019\uff0c\u4f18\u5148\u6dd8\u6c70\u8303\u56f4\u9891\u7387\u6700\u4f4e Key\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5b83\u7684\u5b9e\u73b0\u4e0e LRU \u57fa\u672c\u4e00\u81f4\uff0c\u4f46\u662f\u5728\u8ba1\u6570\u90e8\u5206\u5219\u6709\u6240\u6539\u8fdb\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-1\">3.1 \u6982\u7387\u8ba1\u6570\u5668<\/h4>\n\n\n\n<p>\u5728 Redis \u7528\u6765\u5b58\u50a8\u6570\u636e\u7684\u7ed3\u6784\u4f53 <code>redisObj<\/code> \u4e2d\uff0c\u6709\u4e00\u4e2a 24 \u4f4d\u7684 <code>lru<\/code>\u6570\u503c\u5b57\u6bb5\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u5f53\u4f7f\u7528 LRU \u7b97\u6cd5\u65f6\uff0c\u5b83\u7528\u4e8e\u8bb0\u5f55\u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u7684\u65f6\u95f4\u6233\u3002<\/li>\n\n\n\n<li>\u5f53\u4f7f\u7528 LFU \u7b97\u6cd5\u65f6\uff0c\u5b83\u88ab\u5206\u4e3a\u4e24\u90e8\u5206\uff0c\u9ad8 16 \u4f4d\u5173\u4e8e\u8bb0\u5f55\u6700\u8fd1\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\uff08<code>Last Decrement Time<\/code>\uff09\uff0c\u800c\u4f4e 8 \u4f4d\u4f5c\u4e3a\u8bb0\u5f55\u8bbf\u95ee\u9891\u7387\u8ba1\u6570\u5668\uff08<code>Logistic Counter<\/code>\uff09\u3002<\/li>\n<\/ul>\n\n\n\n<p>LFU \u7684\u6838\u5fc3\u5c31\u5728\u4e8e\u4f4e 8 \u4f4d\u8868\u793a\u7684\u8bbf\u95ee\u9891\u7387\u8ba1\u6570\u5668\uff08\u4e0b\u9762\u6211\u4eec\u7b80\u79f0\u4e3a <code>counter<\/code>\uff09\uff0c\u662f\u4e00\u4e2a\u4ecb\u4e8e 0 ~ 255 \u7684\u7279\u6b8a\u6570\u503c\uff0c\u5b83\u4f1a\u6bcf\u6b21\u8bbf\u95ee Key \u65f6\uff0c\u57fa\u4e8e\u65f6\u95f4\u8870\u51cf\u548c\u6982\u7387\u9012\u589e\u673a\u5236\u52a8\u6001\u6539\u53d8\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>\u8fd9\u79cd\u57fa\u4e8e\u6982\u7387\uff0c\u4f7f\u7528\u6781\u5c0f\u5185\u5b58\u5bf9\u5927\u91cf\u4e8b\u4ef6\u8fdb\u884c\u8ba1\u6570\u7684\u8ba1\u6570\u5668\u88ab\u79f0\u4e3a\u83ab\u91cc\u65af\u8ba1\u6570\u5668\uff0c\u5b83\u662f\u4e00\u79cd\u6982\u7387\u8ba1\u6570\u6cd5\u7684\u5b9e\u73b0\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-2\">3.2 \u65f6\u95f4\u8870\u51cf<\/h4>\n\n\n\n<p>\u6bcf\u5f53\u8bbf\u95ee Key \u65f6\uff0c\u6839\u636e\u5f53\u524d\u5b9e\u9645\u4e0e\u8be5 Key \u7684\u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u65f6\u95f4\u7684\u65f6\u95f4\u5dee\u5bf9 <code>counter<\/code> \u8fdb\u884c\u8870\u51cf\u3002<\/p>\n\n\n\n<p>\u8870\u51cf\u503c\u53d6\u51b3\u4e8e <code>lfu_decay_time<\/code> \u914d\u7f6e\uff0c\u8be5\u914d\u7f6e\u8868\u793a\u4e00\u4e2a\u8870\u51cf\u5468\u671f\u3002\u6211\u4eec\u53ef\u4ee5\u7b80\u5355\u7684\u8ba4\u4e3a\uff0c\u6bcf\u5f53\u65f6\u95f4\u95f4\u9694\u6ee1\u8db3\u4e00\u4e2a\u8870\u51cf\u5468\u671f\u65f6\uff0c\u5c31\u4f1a\u5bf9 <code>counter<\/code> \u51cf\u4e00\u3002<\/p>\n\n\n\n<p>\u6bd4\u5982\uff0c\u6211\u4eec\u8bbe\u7f6e <code>lfu_decay_time<\/code>\u4e3a 1 \u5206\u949f\uff0c\u90a3\u4e48\u5982\u679c Key \u6700\u540e\u4e00\u6b21\u8bbf\u95ee\u8ddd\u79bb\u73b0\u5728\u5df2\u6709 3 \u5206 30 \u79d2\uff0c\u90a3\u4e48 <code>counter<\/code> \u5c31\u9700\u8981\u51cf 3\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-3\">3.3 \u6982\u7387\u9012\u589e<\/h4>\n\n\n\n<p>\u5728\u5b8c\u6210\u8870\u51cf\u540e\uff0cRedis \u5c06\u6839\u636e <code>lfu_log_factor<\/code> \u914d\u7f6e\u5bf9\u5e94\u6982\u7387\u503c\u5bf9 <code>counter<\/code> \u8fdb\u884c\u9012\u589e\u3002<\/p>\n\n\n\n<p>\u8fd9\u91cc\u76f4\u63a5\u653e\u4e0a\u6e90\u7801\uff1a<\/p>\n\n\n\n<p><code>\/* Logarithmically increment a counter. The greater is the current counter value * the less likely is that it gets really implemented. Saturate it at 255. *\/ uint8_t LFULogIncr(uint8_t counter) { &nbsp; &nbsp;\/\/ \u82e5\u5df2\u8fbe\u6700\u5927\u503c 255\uff0c\u76f4\u63a5\u8fd4\u56de &nbsp; &nbsp;if (counter == 255) return 255; &nbsp; &nbsp;\/\/ \u83b7\u53d6\u4e00\u4e2a\u4ecb\u4e8e 0 \u5230 1 \u4e4b\u95f4\u7684\u968f\u673a\u503c &nbsp; &nbsp;double r = (double)rand()\/RAND_MAX; &nbsp; &nbsp;\/\/ \u6839\u636e\u5f53\u524d counter \u51cf\u53bb\u521d\u59cb\u503c\u5f97\u5230 baseval &nbsp; &nbsp;double baseval = counter - LFU_INIT_VAL; &nbsp; &nbsp;if (baseval &lt; 0) baseval = 0; &nbsp; &nbsp;\/\/ \u4f7f\u7528 baseval*server.lfu_log_factor+1 \u5f97\u5230\u4e00\u4e2a\u6982\u7387\u503c p &nbsp; &nbsp;double p = 1.0\/(baseval*server.lfu_log_factor+1); &nbsp; &nbsp;\/\/ \u5f53 r &lt; p \u65f6\uff0c\u9012\u589e counter &nbsp; &nbsp;if (r &lt; p) counter++; &nbsp; &nbsp;return counter; }<\/code><\/p>\n\n\n\n<p>\u7b80\u800c\u8a00\u4e4b\uff0c\u76f4\u63a5\u4ece\u4ee3\u7801\u4e0a\u7406\u89e3\uff0c\u6211\u4eec\u53ef\u4ee5\u8ba4\u4e3a <code>counter<\/code>\u548c <code>lfu_log_factor<\/code> \u8d8a\u5927\uff0c\u5219\u9012\u589e\u7684\u6982\u7387\u8d8a\u5c0f\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"547\" width=\"775\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/0759d52d120737de2a694fab85ab8df2.png\" alt=\"\"><\/p>\n\n\n\n<p>\u5f53\u7136\uff0c\u5b9e\u9645\u4e0a\u4e5f\u8981\u8003\u8651\u5230\u8bbf\u95ee\u6b21\u6570\u5bf9\u5176\u7684\u5f71\u54cd\uff0cRedis \u5b98\u65b9\u7ed9\u51fa\u4e86\u76f8\u5173\u6570\u636e\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"407\" width=\"892\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/7be859390dfd4bdc2171980fd5baec1b.png\" alt=\"\"><\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>\u5177\u4f53\u53ef\u76f4\u63a5\u53c2\u89c1\uff1a<a href=\"https:\/\/www.cnblogs.com\/vivotech\/p\/17531827.html\" target=\"_blank\"  rel=\"nofollow\" >vivo LRU\u4e0eLFU\u7b97\u6cd5\u5b9e\u73b0 sourl.cn\/4yZ6e2<\/a><\/em><\/p>\n<\/blockquote>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-4\">3.4 \u8ba1\u6570\u5668\u7684\u521d\u59cb\u503c<\/h4>\n\n\n\n<p>\u4e3a\u4e86\u9632\u6b62\u65b0\u7684 Key \u7531\u4e8e <code>counter<\/code> \u4e3a 0 \u5bfc\u81f4\u76f4\u63a5\u88ab\u6dd8\u6c70\uff0cRedis \u4f1a\u9ed8\u8ba4\u5c06 <code>counter<\/code>\u8bbe\u7f6e\u4e3a 5\u3002<\/p>\n\n\n\n<h4 class=\"wp-block-heading\" id=\"3-5\">3.5 \u62bd\u6837\u5927\u5c0f\u7684\u9009\u62e9<\/h4>\n\n\n\n<p>\u503c\u5f97\u6ce8\u610f\u7684\u662f\uff0c\u5f53\u6570\u636e\u91cf\u6bd4\u8f83\u5927\u7684\u65f6\u5019\uff0c\u5982\u679c\u62bd\u6837\u5927\u5c0f\u8bbe\u7f6e\u7684\u8fc7\u5c0f\uff0c\u56e0\u4e3a\u4e00\u6b21\u62bd\u6837\u7684\u6837\u672c\u6570\u91cf\u6709\u9650\uff0c\u51b7\u70ed\u6570\u636e\u56e0\u4e3a\u65f6\u95f4\u8870\u51cf\u5bfc\u81f4\u7684\u6743\u91cd\u5dee\u5f02\u5c06\u4f1a\u53d8\u5f97\u4e0d\u660e\u663e\uff0c\u6b64\u65f6 LFU \u7b97\u6cd5\u7684\u4f18\u52bf\u5c31\u96be\u4ee5\u4f53\u73b0\uff0c\u5373\u4f7f\u7684\u76f8\u5bf9\u8f83\u70ed\u7684\u6570\u636e\u4e5f\u6709\u53ef\u80fd\u88ab\u9891\u7e41\u201c\u8bef\u4f24\u201d\u3002<\/p>\n\n\n\n<p>\u6240\u4ee5\uff0c\u5982\u679c\u4f60\u9009\u62e9\u4e86 LFU \u7b97\u6cd5\u4f5c\u4e3a\u6dd8\u6c70\u7b56\u7565\uff0c\u5e76\u4e14\u540c\u65f6\u53c8\u5177\u5907\u6bd4\u8f83\u5927\u7684\u6570\u636e\u91cf\uff0c\u90a3\u4e48\u4e0d\u59a8\u5c06\u62bd\u6837\u5927\u5c0f\u4e5f\u8bbe\u7f6e\u7684\u5927\u4e00\u4e9b\u3002<\/p>\n\n\n\n<p>\u5b8c\u7ed3\uff0c\u6492\u82b1 \ud83c\udf89<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u7b2c19\u5c0f\u8282\uff1a\u5f00\u53d1\u7528\u6237\u4f18\u60e0\u5238\u5206\u53d1\u529f\u80fd\uff08\u4e00\uff09<\/h2>\n\n\n\n<h2 class=\"wp-block-heading\">\u4e1a\u52a1\u80cc\u666f<\/h2>\n\n\n\n<p>\u56e0\u4e3a\u5176\u5b9e\u4f60\u662f\u6709\u4e00\u4e9b\u4fe1\u606f\u7684\u7f3a\u5931\u7684\uff0c\u90a3\u5728\u6211\u5c31\u662f\u4ee5\u6211\u4e2a\u4eba\u7684\u7ecf\u9a8c\u6765\u770b\uff0c\u6211\u4eec\u5728\u5b66\u4e00\u90e8\u5206\u5185\u5bb9\u7684\u65f6\u5019\uff0c\u9996\u5148\u5927\u5bb6\u53ef\u4ee5\u628a\u8fd9\u4e00\u4e2a\u4e8b\u60c5\u7136\u540e\u603b\u7ed3\u6210\u4e00\u4e9b\u6d41\u7a0b\uff0c\u7136\u540e\u6574\u4f53\u7684\u603b\u7ed3\uff0c\u800c\u4e0d\u662f\u4e0a\u6765\u6df1\u5165\u5230\u67d0\u4e9b\u7ec6\u8282\u65b9\u9762\u53bb\uff0c\u5728\u8fd9\u91cc\u7ed9\u5927\u5bb6\u4e00\u4e9b\u5efa\u8bae\uff0c\u5305\u62ec\u6211\u4eec\u770b\u4e00\u4e9b\u6e90\u7801\uff0c\u5305\u62ec\u5b66\u4e60\u4e00\u4e9b\u65b0\u7684\u6280\u672f\u4e5f\u90fd\u662f\u8fd9\u4e2a\u6837\u5b50\u3002<\/p>\n\n\n\n<p>\u5f00\u7bc7\u4e00\u5f20\u56fe\uff0c\u5e2e\u52a9\u5927\u5bb6\u68b3\u7406 v1 \u5206\u53d1\u6d41\u7a0b\u5173\u952e\u8282\u70b9\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full is-resized\"><img loading=\"lazy\" decoding=\"async\" width=\"685\" height=\"629\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759220196-image.png\" alt=\"\" class=\"wp-image-1360\" style=\"width:840px;height:auto\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759220196-image.png 685w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759220196-image-300x275.png 300w\" sizes=\"auto, (max-width: 685px) 100vw, 685px\" \/><\/figure>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u9996\u5148\u901a\u8fc7\u4e00\u4e2a\u65f6\u5e8f\u56fe\u7684\u65b9\u5f0f\uff0c\u5e2e\u5927\u5bb6\u68b3\u7406\u4e00\u4e0b\u6211\u4eec\u5206\u53d1\u7684\u4e00\u4e9b\u5173\u952e\u8282\u70b9\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u53d1\u9001\u4e86\u4e4b\u540e\uff0c\u5b83\u4f1a\u628a\u8fd9\u4e2a\u6d88\u606f\u4e5f\u63a8\u9001\u7ed9\u6d88\u606f\u961f\u5217\u3002\u7136\u540e\u6d88\u606f\u961f\u5217\u7684\u6d88\u8d39\u8005\u662f\u5728\u5206\u53d1\u7cfb\u7edf\u91cc\u9762\uff0c\u76d1\u542c\u5230\u8fd9\u4e2a\u6d88\u606f\u4e4b\u540e\uff0c\u9996\u5148\u6211\u4eec\u4f1a\u5bf9\u5b83\u8fdb\u884c\u4e00\u4e9b\u524d\u7f6e\u6821\u9a8c\u3002\u50cf\u8fd9\u4e00\u6b65\u7684\u8bdd\uff0c\u662f\u5728\u6bcf\u4e2a\u7cfb\u7edf\u91cc\u9762\u90fd\u5fc5\u4e0d\u53ef\u5c11\u7684\u3002\u6bd4\u5982\u8bf4\uff0c\u5982\u679c\u6211\u4eec\u662f\u5b9a\u65f6\u7684\u53d1\u9001\uff0c\u5b83\u6709\u53ef\u80fd\u7528\u6237\u8fd0\u8425\u4eba\u5458\u89c9\u5f97\u201c\u6211\u4e0d\u9700\u8981\u4e86\u201d\uff0c\u5c31\u628a\u63a8\u9001\u4efb\u52a1\u53d6\u6d88\u4e86\u3002\u5982\u679c\u4f60\u6ca1\u6709\u5224\u65ad\u4efb\u52a1\u7684\u4e00\u4e9b\u72b6\u6001\u5c31\u53bb\u6267\u884c\uff0c\u5176\u5b9e\u7cfb\u7edf\u5c31\u6709\u95ee\u9898\u3002\u5305\u62ec\uff1a\u8fd9\u4e2a\u6a21\u677f\u662f\u4e0d\u662f\u5df2\u7ecf\u5230\u671f\u4e86\uff1f\u518d\u6216\u8005\u8bf4\uff0c\u8fd9\u4e2a\u6a21\u677f\u7684\u4f18\u60e0\u5238\u6570\u91cf\u8fd8\u6709\u6ca1\u6709\uff1f\u7b49\u7b49\uff0c\u8fd9\u4e9b\u6211\u4eec\u90fd\u8981\u5f3a\u5236\u53bb\u8fdb\u884c\u4e00\u4e9b\u5224\u65ad\u3002<\/p>\n\n\n\n<p>\u518d\u7136\u540e\u7684\u8bdd\uff0c\u6211\u4eec\u672c\u8d28\u4e0a\u662f\u901a\u8fc7\u4f18\u60e0\u5238\u5206\u53d1\uff0c\u662f\u901a\u8fc7 Excel \u53bb\u8fdb\u884c\u5904\u7406\u7684\u3002\u6211\u4eec\u53bb\u8bfb\u53d6 Excel \u7684\u65f6\u5019\uff0c\u5b83\u7684\u6bcf\u4e00\u884c\u6211\u4eec\u90fd\u8981\u505a\u4e00\u4e9b\u5904\u7406\u3002\u6211\u4eec\u9996\u5148\u8981\u5224\u65ad\u6570\u636e\u5e93\u91cc\u9762\u662f\u5426\u6709\u6211\u4eec\u5bf9\u5e94\u7684\u5e93\u5b58\uff0c\u5305\u62ec\u7f13\u5b58\u3002\u7136\u540e\uff0c\u5982\u679c\u5e93\u5b58\u5145\u8db3\u7684\u60c5\u51b5\u4e0b\uff0c\u6211\u4eec\u4f1a\u53bb\u5728\u7f13\u5b58\u548c\u6570\u636e\u5e93\u91cc\u9762\u8fdb\u884c\u76f8\u5e94\u7684\u6263\u51cf\uff0c\u5e76\u4e14\u5c06\u7528\u6237\u7684\u9886\u5238\u8bb0\u5f55\u6dfb\u52a0\u5230\u6211\u4eec\u7684\u7f13\u5b58\u5f53\u4e2d\u3002\u56e0\u4e3a\u540e\u7eed\u7528\u6237\u53bb\u4f7f\u7528\u65f6\uff0c\u6211\u4eec\u80af\u5b9a\u4e0d\u80fd\u8ba9\u7528\u6237\u76f4\u63a5\u53bb\u67e5\u6570\u636e\u5e93\uff0c\u5bf9\u6570\u636e\u5e93\u7684\u538b\u529b\u4f1a\u6709\u4e00\u70b9\u5927\u3002\u6240\u4ee5\u9996\u5148\u662f\u52a0\u5230\u7f13\u5b58\u91cc\u9762\u3002<\/p>\n\n\n\n<p>\u5982\u679c\u8bf4\u5f53\u5e93\u5b58\u4e0d\u8db3\uff0c\u6216\u8005\u8bf4\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u5b8c\u4f18\u60e0\u5238\u7684\u60c5\u51b5\u4e0b\uff0c\u90a3\u4e48\u6211\u4eec\u672c\u6761\u8bb0\u5f55\u8df3\u8fc7\u5373\u53ef\u3002<\/p>\n\n\n\n<p>\u6700\u540e\uff0c\u5982\u679c\u6211\u4eec\u5224\u65ad Excel \u5df2\u7ecf\u5230\u4e86\u6700\u540e\u4e00\u884c\u4e86\uff0c\u90a3\u5c31\u8bc1\u660e\u6211\u4eec\u7684\u4efb\u52a1\u5df2\u7ecf\u7ed3\u675f\u4e86\u3002\u6211\u4eec\u5c31\u8981\u5bf9\u5206\u53d1\u7684\u4efb\u52a1\u8bb0\u5f55\u53bb\u4fee\u6539\u5b83\u7684\u4e00\u4e2a\u4efb\u52a1\u72b6\u6001\uff0c\u6bd4\u5982\u8bf4\u4ece\u201c\u6267\u884c\u4e2d\u201d\u53d8\u6210\u201c\u7ed3\u675f\u201d\uff0c\u4ee5\u53ca\u6211\u4eec\u4ec0\u4e48\u65f6\u5019\u7ed3\u675f\u7684\uff0c\u628a\u5b83\u7684\u5b8c\u6210\u65f6\u95f4\u4e5f\u66f4\u65b0\u4e00\u4e0b\u3002<\/p>\n\n\n\n<p>\u6700\u540e\uff0c\u6211\u4eec\u628a\u5bf9\u5e94\u7684\u6d88\u606f\u961f\u5217\u7684\u4e00\u4e2a ACK \u54cd\u5e94\u8fd4\u56de\u7ed9\u6d88\u606f\u961f\u5217\uff0c\u6211\u4eec\u6574\u4e2a\u6d41\u7a0b\u5c31\u7ed3\u675f\u4e86\u3002\u5176\u5b9e\u8fd9\u6837\u770b\u8d77\u6765\u662f\u6bd4\u8f83\u7b80\u5355\u7684\u2014\u2014\u5f53\u7136\u4e5f\u662f\u56e0\u4e3a\u6211\u4eec V1 \u8282\u70b9\u5b83\u7684\u6d41\u7a0b\u4f1a\u504f\u7b80\u5355\u4e00\u70b9\uff0c\u6240\u4ee5\u8bf4\u8fd9\u4e48\u770b\u8fd8\u597d\u3002<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n<\/blockquote>\n\n\n\n<p>\u672c\u7bc7\u6587\u7ae0\u66f4\u591a\u805a\u7126\u5728\u5206\u53d1\u6d88\u606f\u961f\u5217\u6d88\u8d39\u8005\uff0c\u5728\u6267\u884c\u6d88\u606f\u6d88\u8d39\u65f6\uff0c\u5e94\u8be5\u9996\u5148\u8bfb\u53d6 Excel \u6570\u636e\uff0c\u7136\u540e\u6309\u7167\u7528\u6237\u5206\u53d1\u6d41\u7a0b\u8fdb\u884c\u64cd\u4f5c\u3002<\/p>\n\n\n\n<p>\u5177\u4f53\u7684\u5206\u53d1\u903b\u8f91\u5982\u4e0b\uff1a<\/p>\n\n\n\n<ol class=\"wp-block-list\">\n<li>1.\u68c0\u67e5\u4f18\u60e0\u5238\u6a21\u677f\u7684\u72b6\u6001\u662f\u5426\u6b63\u5e38\uff1b<\/li>\n\n\n\n<li>2.\u9a8c\u8bc1\u4f18\u60e0\u5238\u6a21\u677f\u7684\u5e93\u5b58\u4f59\u91cf\u662f\u5426\u5145\u8db3\uff1b<\/li>\n\n\n\n<li>3.\u786e\u8ba4\u7528\u6237\u662f\u5426\u5df2\u9886\u53d6\u76f8\u540c\u7684\u4f18\u60e0\u5238\uff0c\u82e5\u5df2\u9886\u53d6\u5219\u4e0d\u518d\u5206\u53d1\uff1b<\/li>\n\n\n\n<li>4.\u8bb0\u5f55\u7528\u6237\u7684\u9886\u5238\u4fe1\u606f\u3002<\/li>\n<\/ol>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>\u6ce8\u610f\uff0c\u5982\u679c\u524d\u9762\u5546\u5bb6\u540e\u7ba1\u670d\u52a1\u6dfb\u52a0\u4e86 vm \u53c2\u6570\uff0c\u5206\u53d1\u670d\u52a1\u8bb0\u5f97\u4e5f\u52a0\u4e0a\uff0c\u8981\u4e0d\u7136\u6d88\u8d39\u4e0d\u5230\u5bf9\u5e94\u6d88\u606f\u961f\u5217\u6d88\u606f\u3002<\/em>\u540e\u7ba1\u7684\u8bdd\uff0c\u4f60\u770b\u5728\u8fd9\u91cc\u9762\u662f\u6709\u4e00\u4e9b VM \u53c2\u6570\u7684\u3002\u5f53\u7136\u6211\u81ea\u5df1\u5728\u8054\u8c03\u7684\u65f6\u5019\uff0c\u6211\u5176\u5b9e\u662f\u6709\u4e00\u4e9b\u6539\u52a8\u7684\uff0c\u5927\u5bb6\u6309\u7167\u4e4b\u524d\u6211\u4eec\u5728\u6587\u6863\u91cc\u9762\u5199\u7684\u5c31\u53ef\u4ee5\u4e86\u3002\u7136\u540e\u6211\u4eec\u8fd9\u6b21\u8d77\u6765\u4e86\uff0c\u5206\u53d1\u670d\u52a1\u5176\u5b9e\u4e5f\u8981\u628a\u5bf9\u5e94\u7684\u4e00\u4e9b\u4e1c\u897f\u7ed9\u5b83\u52a0\u4e0a<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"759\" height=\"414\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759307718-image.png\" alt=\"\" class=\"wp-image-1368\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759307718-image.png 759w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759307718-image-300x164.png 300w\" sizes=\"auto, (max-width: 759px) 100vw, 759px\" \/><\/figure>\n<\/blockquote>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"git\">Git \u5206\u652f<\/h2>\n\n\n\n<p>20240829_dev_coupon-distribute-v1_easyexcel-cache_ding.ma<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6570\u636e\u5e93\u8868\u8bbe\u8ba1<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u7528\u6237\u4f18\u60e0\u5238\u8868\u8bbe\u8ba1<\/h3>\n\n\n\n<p>\u7528\u6237\u4f18\u60e0\u5238\u8868\u8bbe\u8ba1\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>CREATE TABLE `t_user_coupon` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';<\/code><\/pre>\n\n\n\n<p>\u5176\u4e2d\u6709\u4e00\u4e2a\u5b57\u6bb5\u7528\u4e8e\u8bb0\u5f55\u7528\u6237\u9886\u53d6\u4f18\u60e0\u5238\u7684\u6b21\u6570\uff0c\u4ee5\u6807\u8bc6\u8fd9\u662f\u7528\u6237\u7b2c\u51e0\u6b21\u9886\u53d6\u3002\u901a\u8fc7\u8bbe\u7f6e\u552f\u4e00\u7d22\u5f15\u6765\u9650\u5236\u7528\u6237\u7684\u9886\u53d6\u6b21\u6570\u3002\u56e0\u4e3a\u7528\u6237\u4ed6\u9886\u53d6\u7684\u4f18\u60e0\u5238\uff0c\u6700\u7ec8\u662f\u8981\u5728\u6570\u636e\u5e93\u8868\u91cc\u9762\u8fdb\u884c\u4e00\u4e2a\u4fdd\u5b58\u7684\u3002\u6bd4\u5982\u8bf4\u8fd9\u91cc\u9762\u4f1a\u8bb0\u5f55\u5b83\u7684\u9886\u53d6\u65f6\u95f4\uff0c\u4ee5\u53ca\u5b83\u7684\u9886\u53d6\u6b21\u6570\u2014\u2014\u56e0\u4e3a\u5927\u5bb6\u4e4b\u524d\u5728\u5b66\u4f18\u60e0\u5238\u6a21\u677f\u7684\u65f6\u5019\uff0c\u5176\u5b9e\u5b83\u91cc\u9762\u662f\u6709\u9650\u5236\u7528\u6237\u80fd\u591f\u9886\u53d6\u591a\u5c11\u6b21\u7684\uff0c\u8fd9\u4e2a\u9650\u5236\u5c31\u901a\u8fc7\u6211\u4eec\u8fd9\u91cc\u53bb\u8fdb\u884c\u7684\u4e00\u4e2a\u6821\u9a8c\u3002\u7136\u540e\u8fd8\u6709\u6709\u6548\u671f\u7684\u5f00\u59cb\u548c\u7ed3\u675f\u65f6\u95f4\uff0c\u4ee5\u53ca\u5b83\u5982\u679c\u4f7f\u7528\u7684\u8bdd\u4e00\u4e2a\u4f7f\u7528\u65f6\u95f4\uff0c\u8fd8\u6709\u5b83\u7684\u4e00\u4e2a\u6765\u6e90\u4ee5\u53ca\u5b83\u7684\u4e00\u4e2a\u72b6\u6001\u3002\u8fd9\u4e2a\u7684\u8bdd\u4f1a\u5728\u6211\u4eec\u540e\u9762\u7ae0\u8282\u4f1a\u7528\u5230\uff0c\u9ed8\u8ba4\u7684\u8bdd\u9886\u53d6\u5b8c\u4e4b\u540e\u662f\u4e00\u4e2a\u201c\u672a\u4f7f\u7528\u201d\u7684\u72b6\u6001\u3002\u7136\u540e\u6211\u4eec\u8fd9\u91cc\u9762\u5176\u5b9e\u8fd8\u6709\u4e00\u4e9b\u552f\u4e00\u7d22\u5f15\uff0c\u4e5f\u5c31\u662f\u521a\u624d\u6211\u4eec\u8bb2\u7684\u9886\u53d6\u6b21\u6570\uff0c\u5728\u6570\u636e\u5e93\u505a\u4e00\u5c42\u515c\u5e95\u3002<\/p>\n\n\n\n<p>\u7528\u6237\u4f18\u60e0\u5238\u9886\u53d6\u2014\u2014\u5927\u5bb6\u90fd\u7528\u8fc7\u4e00\u4e9b\u50cf\u7f8e\u56e2\u3001\u997f\u4e86\u4e48\u3001\u4eac\u4e1c\u8fd9\u4e9b\u5e73\u53f0\u2014\u2014\u5b83\u5176\u5b9e\u7528\u6237\u7684\u9886\u5238\u91cf\uff0c\u5b83\u7684\u6570\u636e\u91cf\u662f\u5f88\u5927\u7684\u3002\u6240\u4ee5\u8bf4\u6211\u4eec\u8981\u5bf9\u5e93\u8fdb\u884c\u5206\u5e93\u5206\u8868\u7684\u4e00\u4e2a\u5904\u7406\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u4e00\u822c\u6765\u8bb2\uff0c\u50cf\u8fd9\u4e2a\u91cf\u7ea7\u6216\u8005\u8bf4\u5b83\u7684\u8fd9\u79cd\u6570\u636e\u590d\u6742\u5ea6\uff0c\u8bf4\u53e5\u5b9e\u5728\u8bdd\uff0c\u53ea\u8981\u4f60\u6ca1\u6709\u8fdb\u884c\u590d\u6742\u7684\u90a3\u4e9b\u5206\u7ec4\u6392\u5e8f\u7b49\u903b\u8f91\uff0c\u54ea\u6015\u6570\u636e\u5e93\u8868\u91cc\u9762\u5b83\u662f\u4e00\u4e2a\u4ebf\u7ea7\uff0c\u6bd4\u5982\u8bf4 1 \u4ebf\uff0c\u5bf9\u5427\uff1f\u5176\u5b9e\u6ca1\u6709\u201c\u4e00\u822c\u6765\u8bf4\u201d\uff0c\u6211\u4eec\u4f1a\u53d6\u4e00\u4e2a\u8bd5\u7528\u503c\uff0c\u8fd9\u4e2a\u662f\u6ca1\u6709\u7edd\u5bf9\u7684\u597d\u5427\uff1f<\/p>\n\n\n\n<p>\u6211\u4e0a\u9762\u8bf4\u7684\u662f\u53d6\u51b3\u4e8e\u4e0d\u540c\u7cfb\u7edf\u7684\u4e00\u4e2a\u91cf\u7ea7\u3002\u8fd9\u4e2a\u4e0d\u662f\u6211\u4eec\u5728\u8fd9\u91cc\u53bb\u9884\u4f30\u7684\uff0c\u800c\u662f\u8bf4\u5728\u7cfb\u7edf\u7684\u8fd0\u884c\u8fc7\u7a0b\u5f53\u4e2d\u4f60\u8981\u53bb\u8bc4\u4f30\uff0c\u7136\u540e\u624d\u80fd\u53bb\u8fdb\u884c\u5206\u8868\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u9762\u8bd5\u5b98\u6211\u4eec\u5728\u548c\u4ed6\u804a\u7684\u65f6\u5019\uff0c\u6211\u4eec\u5c31\u53ef\u4ee5\u8bf4\uff1a\u201c\u5728\u672a\u6765\u591a\u5c11\u5e74\uff0c\u6211\u4eec\u7684\u6570\u636e\u91cf\u5927\u6982\u63a8\u4f30\u6709\u591a\u5c11\uff0c\u7136\u540e\u6839\u636e\u6211\u4eec\u8868\u7684\u4e00\u4e2a\u590d\u6742\u5ea6\uff0c\u6211\u4eec\u6700\u7ec8\u786e\u5b9a\u4e3a\u591a\u5c11\u5f20\u8868\u3002\u201d\u8fd9\u91cc\u6211\u76f4\u63a5\u8bbe\u7f6e\u4e86 32 \u5f20\u8868\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u521a\u624d\u6211\u4eec\u8bb2\u5230\u6211\u4eec\u9700\u8981\u5206\u5e93\u5206\u8868\u3002\u6570\u636e\u91cf\u5927\uff0c\u76f4\u63a5\u5206\u8868\u5c31\u884c\u4e86\u3002\u4f46\u662f\u6211\u4eec\u8fd9\u91cc\u4e3a\u4ec0\u4e48\u8981\u8003\u8651\u5206\u5e93\uff1f\u5206\u5e93\u662f\u56e0\u4e3a\u5bf9\u6570\u636e\u5e93\u7528\u6237\u9886\u53d6\u7684\u8bdd\uff0c\u5b83\u7684\u5bf9\u6570\u636e\u5e93\u7684\u64cd\u4f5c\u8fd8\u662f\u86ee\u9891\u7e41\u7684\u3002\u5206\u8868\u662f\u4e3b\u8981\u89e3\u51b3\u6570\u636e\u91cf\u5927\u3001\u67e5\u8be2\u6162\u7684\u95ee\u9898\uff1b\u5206\u5e93\u7684\u8bdd\uff0c\u5c31\u662f\u89e3\u51b3\u5e76\u53d1\u5927\u3001\u7136\u540e\u5355\u5e93\u625b\u4e0d\u4f4f\u7684\u95ee\u9898\uff1b\u56e0\u4e3a\u6211\u4eec\u7684\u5206\u914d\u7b56\u7565\u548c\u4f18\u60e0\u5238\u6a21\u677f\u57fa\u672c\u4e0a\u662f\u76f8\u540c\u7684\uff0c\u6240\u4ee5\u8bf4\u6211\u4eec\u8fd9\u91cc\u9762\u76f4\u63a5\u53bb\u52a0\u5c31\u597d\u4e86\u3002\u7136\u540e\u8fd9\u662f\u6211\u4eec\u7684\u7528\u6237\u4f18\u60e0\u5238\u8868\uff0c\u7136\u540e\u53ef\u4ee5\u770b\u5230\u90fd\u662f\u91c7\u7528\u54c8\u5e0c\u7684\u65b9\u5f0f\uff0c\u5927\u5bb6\u5e94\u8be5\u6bd4\u8f83\u719f\u6089\u4e86\u3002\u7136\u540e\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u53bb\u5bf9\u5e94\u7684\u6570\u636e\u5e93\uff0c\u521b\u5efa\u6211\u4eec\u5bf9\u5e94\u7684\u6570\u636e\u5e93\u8868<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u7528\u6237\u4f18\u60e0\u5238\u5206\u5e93\u5206\u8868\u8bbe\u8ba1<\/h3>\n\n\n\n<p>\u7531\u4e8e\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\u7684\u6570\u636e\u91cf\u975e\u5e38\u5927\uff0c\u9700\u8981\u5bf9\u8be5\u8868\u8fdb\u884c\u5206\u5e93\u5206\u8868\u5904\u7406\u3002\u4e0d\u540c\u4e8e\u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u53ef\u4ee5\u63a8\u884d\u4f30\u7b97\uff0c\u6211\u4eec\u5c06\u76f4\u63a5\u5206\u4e3a 32 \u5f20\u8868\uff0c\u4ee5\u786e\u4fdd\u5355\u8868\u6570\u636e\u91cf\u4fdd\u6301\u5728\u4ebf\u7ea7\u4ee5\u4e0b\uff0c\u4ece\u800c\u907f\u514d\u6027\u80fd\u538b\u529b\u3002<\/p>\n\n\n\n<p>\u8003\u8651\u5230\u5927\u91cf\u7528\u6237\u9891\u7e41\u9886\u53d6\u4f18\u60e0\u5238\u5bfc\u81f4\u5199\u5165\u64cd\u4f5c\u8f83\u4e3a\u9891\u7e41\uff0c\u56e0\u6b64\u4e5f\u9700\u8981\u8fdb\u884c\u5206\u5e93\u3002\u5206\u7247\u7b56\u7565\u4e0e\u4f18\u60e0\u5238\u6a21\u677f\u76f8\u540c\uff0c\u91c7\u7528 Hash \u65b9\u5f0f\u8fdb\u884c\u5206\u5e93\u5206\u8868\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u7136\u540e\u5927\u5bb6\u8fd9\u91cc\u9762\u6709\u4e24\u70b9\u6ce8\u610f\u4e8b\u9879\uff1a<\/p>\n\n\n\n<p>\u9996\u5148\u7b2c\u4e00\u70b9\u7684\u8bdd\uff0c\u5c31\u662f\u8fd9\u4e2a\u7ed3\u675f\u65f6\u95f4\u3002\u56e0\u4e3a\u6211\u4eec\u73b0\u5728\u662f 25 \u5e74\u7684 6 \u6708\u4efd\uff0c\u6240\u4ee5\u8bf4\u5982\u679c\u662f 25 \u5e74\u9ed8\u8ba4\u65f6\u95f4\u597d\u50cf\u662f 25 \u5e74 9 \u6708\u4efd\u2014\u2014\u8fd9\u4e2a\u662f\u6211\u5f53\u65f6\u968f\u4fbf\u5199\u7684\uff0c26 \u5e74\u4e4b\u524d\u90fd\u53ef\u4ee5\u3002\u5927\u5bb6\u5728\u5b66\u7684\u65f6\u5019\uff0c\u5982\u679c\u53d1\u73b0\u5f53\u524d\u7684\u65f6\u95f4\u5df2\u7ecf\u5927\u4e8e\u8fd9\u4e2a\u65f6\u95f4\uff0c\u8bf7\u5927\u5bb6\u6539\u4e00\u4e0b\uff0c\u8981\u4e0d\u7136\u7684\u8bdd\u4f60\u521b\u5efa\u5b8c\u6a21\u677f\u7684\u65f6\u5019\u5b83\u5c31\u4f1a\u88ab\u8fc7\u671f\u3002\u56e0\u4e3a\u6211\u4eec\u4f18\u60e0\u5238\u6a21\u677f\u8fc7\u671f\u4e86\u4e4b\u540e\u662f\u81ea\u52a8\u53bb\u7ed9\u5b83\u9500\u6bc1\u7684\uff0c\u597d\u5427\uff1f<\/p>\n\n\n\n<p>\u7136\u540e\u7b2c\u4e8c\u4e2a\u7684\u8bdd\uff0c\u5c31\u662f\u5b83\u7684\u5e93\u5b58\u91cf\u3002\u56e0\u4e3a\u6211\u4eec\u63a5\u4e0b\u6765\u6211\u4eec\u53bb\u6267\u884c\uff0c\u6211\u4eec\u5e94\u8be5\u662f 5000 \u6761\u8bb0\u5f55\uff0c\u8fd9\u91cc\u6211\u8bbe\u7f6e 5001 \u7684\u4e00\u4e2a\u5e93\u5b58\uff0c\u65b9\u4fbf\u5927\u5bb6\u53bb\u770b\u4e00\u4e0b\u3002\u597d\u5427\uff0c\u8fd9\u91cc\u7684\u8bdd\u6211\u4eec\u76f4\u63a5\u6267\u884c\u3002<\/p>\n\n\n\n<p>\u6267\u884c\u5b8c\u4e86\u4e4b\u540e\uff0c\u6211\u4eec\u53bb\u770b\u4e00\u4e0b\u6211\u4eec\u7684\u6570\u636e\u5e93\u8868\u3002\u5982\u679c\u6309\u7167\u6211\u4eec\u7684\u5206\u7247\u89c4\u5219\uff0c\u6211\u4eec\u7684\u4f18\u60e0\u5238\u6a21\u677f\u5e94\u8be5\u5728\u6211\u4eec\u7684\u4e00\u5e93\u548c\u7b2c\u5341\u4e94\u5f20\u8868\u91cc\u9762\u3002\u6211\u4eec\u628a ID \u7ed9\u590d\u5236\u51fa\u6765\uff0c\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\u6211\u4eec\u7684\u5e93\u5b58\u662f 5001\u3002\u7136\u540e\u6211\u4eec\u76f8\u540c\u7684\u5176\u5b9e\u662f\u76f8\u540c\u7684\u662f\u6a21\u677f\u3002<\/p>\n\n\n\n<p>\u8fd9\u4e2a\u662f\u6211\u4eec\u5bf9\u5e94\u7684\u4f18\u60e0\u5238\u63a8\u9001\u2014\u2014\u521b\u5efa\u5206\u53d1\u4efb\u52a1\u8fd9\u4e00\u5757\u3002\u628a\u4f18\u60e0\u5238 ID \u6539\u4e00\u4e0b\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u8fd9\u91cc\u9762\u521a\u624d\u6211\u4eec\u6709\u8bf4\uff0c\u5b83\u662f\u6709\u4e24\u79cd\u53d1\u9001\u65b9\u5f0f\uff1a\u7b2c\u4e00\u79cd\u7684\u8bdd\u5c31\u662f\u7acb\u5373\u53d1\u9001\uff0c\u5bf9\u5427\uff1f\u7b2c\u4e8c\u79cd\u7684\u8bdd\u662f\u5b9a\u65f6\u53d1\u9001\u3002\u5982\u679c\u662f\u5b9a\u65f6\u53d1\u9001\u7684\u8bdd\uff0c\u8fd9\u91cc type \u662f 1\uff0c\u7136\u540e\u4e0b\u9762\u8fd8\u6709\u4e00\u4e2a systemTime\u3002\u8fd9\u91cc\u6211\u4eec\u5c31\u4e0d\u6f14\u793a\u5b9a\u65f6\u4e86\uff0c\u8fd9\u4e2a\u529f\u80fd\u80af\u5b9a\u662f\u597d\u7684\uff0c\u7136\u540e\u6211\u4eec\u8fd9\u91cc\u9762\u76f4\u63a5\u5b9e\u65f6\u63a8\u9001\u5c31\u597d\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"857\" height=\"556\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759308981-image.png\" alt=\"\" class=\"wp-image-1369\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759308981-image.png 857w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759308981-image-300x195.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759308981-image-768x498.png 768w\" sizes=\"auto, (max-width: 857px) 100vw, 857px\" \/><\/figure>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u8fd9\u91cc\u9762\u2026\u2026\u6211\u770b\u770b\u6709\u6ca1\u6709\u6253debug\uff1f\u7a0d\u7b49\u3002\u5728\u7b2c\u4e00\u6b65\u7684\u8bdd\uff0c\u5b83\u5e94\u8be5\u662f\u5728 service \u91cc\u9762\u6709\u7684\u3002\u7136\u540e\u7b2c\u4e8c\u6b65\u7684\u8bdd\u5e94\u8be5\u662f\u5728\u6211\u4eec\u7684controller \u91cc\u9762\u3002OK\uff0c\u6ca1\u95ee\u9898\uff0c\u7136\u540e\u6211\u4eec\u70b9\u4e00\u4e0b\u53d1\u9001\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"844\" height=\"734\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759309009-image.png\" alt=\"\" class=\"wp-image-1370\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759309009-image.png 844w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759309009-image-300x261.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759309009-image-768x668.png 768w\" sizes=\"auto, (max-width: 844px) 100vw, 844px\" \/><\/figure>\n\n\n\n<p>\u7136\u540e\u5728\u8fd9\u91cc\u9762\uff0c\u56e0\u4e3a\u5224\u65ad\u6211\u4eec\u7684\u7c7b\u578b\u7684\u8bdd\u5b83\u662f 0\uff0c\u6240\u4ee5\u8bf4\u662f\u7acb\u5373\u53d1\u9001\u3002\u6211\u4eec\u901a\u8fc7\u6d88\u606f\u961f\u5217\u7684\u65b9\u5f0f\uff0c\u7136\u540e\u53d1\u9001\u7ed9\u6211\u4eec\u5bf9\u5e94\u7684\u5206\u53d1\u670d\u52a1\u3002\u7136\u540e\u8fd9\u91cc\u9762\u5c31\u5230\u8fd9\u91cc\u4e86\u3002<\/p>\n<\/blockquote>\n\n\n\n<pre class=\"wp-block-code\"><code># \u6570\u636e\u6e90\u96c6\u5408\ndataSources:\n  # \u81ea\u5b9a\u4e49\u6570\u636e\u6e90\u540d\u79f0\uff0c\u53ef\u4ee5\u662f ds_0 \u4e5f\u53ef\u4ee5\u53eb datasource_0 \u90fd\u53ef\u4ee5\n  ds_0:\n    dataSourceClassName: com.zaxxer.hikari.HikariDataSource\n    driverClassName: com.mysql.cj.jdbc.Driver\n    jdbcUrl: jdbc:mysql:\/\/127.0.0.1:3306\/one_coupon_rebuild_0?useUnicode=true&amp;characterEncoding=UTF-8&amp;rewriteBatchedStatements=true&amp;allowMultiQueries=true&amp;serverTimezone=Asia\/Shanghai\n    username: root\n    password: root\n  ds_1:\n    dataSourceClassName: com.zaxxer.hikari.HikariDataSource\n    driverClassName: com.mysql.cj.jdbc.Driver\n    jdbcUrl: jdbc:mysql:\/\/127.0.0.1:3306\/one_coupon_rebuild_1?useUnicode=true&amp;characterEncoding=UTF-8&amp;rewriteBatchedStatements=true&amp;allowMultiQueries=true&amp;serverTimezone=Asia\/Shanghai\n    username: root\n    password: root\n\nrules:\n  - !SHARDING\n    tables: # \u9700\u8981\u5206\u7247\u7684\u6570\u636e\u5e93\u8868\u96c6\u5408\n      t_coupon_template: # \u4f18\u60e0\u5238\u6a21\u677f\u8868\n        # \u771f\u5b9e\u5b58\u5728\u6570\u636e\u5e93\u4e2d\u7684\u7269\u7406\u8868\n        actualDataNodes: ds_${0..1}.t_coupon_template_${0..15}\n        databaseStrategy: # \u5206\u5e93\u7b56\u7565\n          standard: # \u5355\u5206\u7247\u952e\u5206\u5e93\n            shardingColumn: shop_number # \u5206\u7247\u952e\n            shardingAlgorithmName: coupon_template_database_mod # \u5e93\u5206\u7247\u7b97\u6cd5\u540d\u79f0\uff0c\u5bf9\u5e94 rules&#91;0].shardingAlgorithms\n        tableStrategy: # \u5206\u8868\u7b56\u7565\n          standard: # \u5355\u5206\u7247\u952e\u5206\u8868\n            shardingColumn: shop_number # \u5206\u7247\u952e\n            shardingAlgorithmName: coupon_template_table_mod # \u8868\u5206\u7247\u7b97\u6cd5\u540d\u79f0\uff0c\u5bf9\u5e94 rules&#91;0].shardingAlgorithms\n      t_user_coupon:\n        actualDataNodes: ds_${0..1}.t_user_coupon_${0..31}\n        databaseStrategy:\n          standard:\n            shardingColumn: user_id\n            shardingAlgorithmName: user_coupon_database_mod\n        tableStrategy:\n          standard:\n            shardingColumn: user_id\n            shardingAlgorithmName: user_coupon_table_mod\n    shardingAlgorithms: # \u5206\u7247\u7b97\u6cd5\u5b9a\u4e49\u96c6\u5408\n      coupon_template_database_mod: # \u4f18\u60e0\u5238\u5206\u5e93\u7b97\u6cd5\u5b9a\u4e49\n        type: CLASS_BASED # \u6839\u636e\u81ea\u5b9a\u4e49\u5e93\u5206\u7247\u7b97\u6cd5\u7c7b\u8fdb\u884c\u5206\u7247\n        props: # \u5206\u7247\u76f8\u5173\u5c5e\u6027\n          # \u81ea\u5b9a\u4e49\u5e93\u5206\u7247\u7b97\u6cd5Class\n          algorithmClassName: com.nageoffer.onecoupon.distribution.dao.sharding.DBHashModShardingAlgorithm\n          sharding-count: 16 # \u5206\u7247\u603b\u6570\u91cf\n          strategy: standard # \u5206\u7247\u7c7b\u578b\uff0c\u5355\u5b57\u6bb5\u5206\u7247\n      coupon_template_table_mod: # \u4f18\u60e0\u5238\u5206\u8868\u7b97\u6cd5\u5b9a\u4e49\n        type: CLASS_BASED # \u6839\u636e\u81ea\u5b9a\u4e49\u5e93\u5206\u7247\u7b97\u6cd5\u7c7b\u8fdb\u884c\u5206\u7247\n        props: # \u5206\u7247\u76f8\u5173\u5c5e\u6027\n          # \u81ea\u5b9a\u4e49\u8868\u5206\u7247\u7b97\u6cd5Class\n          algorithmClassName: com.nageoffer.onecoupon.distribution.dao.sharding.TableHashModShardingAlgorithm\n          strategy: standard # \u5206\u7247\u7c7b\u578b\uff0c\u5355\u5b57\u6bb5\u5206\u7247\n      user_coupon_database_mod:\n        type: CLASS_BASED\n        props:\n          algorithmClassName: com.nageoffer.onecoupon.distribution.dao.sharding.DBHashModShardingAlgorithm\n          sharding-count: 32\n          strategy: standard\n      user_coupon_table_mod:\n        type: CLASS_BASED\n        props:\n          algorithmClassName: com.nageoffer.onecoupon.distribution.dao.sharding.TableHashModShardingAlgorithm\n          strategy: standard\n\nprops:\n  # \u914d\u7f6e ShardingSphere \u9ed8\u8ba4\u6253\u5370 SQL \u6267\u884c\u8bed\u53e5\n  sql-show: true<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3-sql\">3. \u521b\u5efa\u7528\u6237\u4f18\u60e0\u5238\u8868 SQL<\/h3>\n\n\n\n<p>\u8fdb\u5165 <code>one_coupon_rebuild_0<\/code> \u6570\u636e\u5e93\uff0c\u6267\u884c\u4e0b\u8ff0 SQL\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>CREATE TABLE `t_user_coupon_0` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_1` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_2` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_3` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_4` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_5` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_6` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_7` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_8` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_9` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_10` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_11` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_12` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_13` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_14` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_15` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';<\/code><\/pre>\n\n\n\n<p>\u8fdb\u5165 <code>one_coupon_rebuild_1<\/code> \u6570\u636e\u5e93\uff0c\u6267\u884c\u4e0b\u8ff0 SQL\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>CREATE TABLE `t_user_coupon_16` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_17` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_18` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_19` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_20` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_21` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_22` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_23` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_24` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_25` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_26` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_27` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_28` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_29` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_30` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';\n\nCREATE TABLE `t_user_coupon_31` (\n  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n  `user_id` bigint(20) DEFAULT NULL COMMENT '\u7528\u6237ID',\n  `coupon_template_id` bigint(20) DEFAULT NULL COMMENT '\u4f18\u60e0\u5238\u6a21\u677fID',\n  `receive_time` datetime DEFAULT NULL COMMENT '\u9886\u53d6\u65f6\u95f4',\n  `receive_count` int(3) DEFAULT NULL COMMENT '\u9886\u53d6\u6b21\u6570',\n  `valid_start_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u5f00\u59cb\u65f6\u95f4',\n  `valid_end_time` datetime DEFAULT NULL COMMENT '\u6709\u6548\u671f\u7ed3\u675f\u65f6\u95f4',\n  `use_time` datetime DEFAULT NULL COMMENT '\u4f7f\u7528\u65f6\u95f4',\n  `source` tinyint(1) DEFAULT NULL COMMENT '\u5238\u6765\u6e90 0\uff1a\u9886\u5238\u4e2d\u5fc3 1\uff1a\u5e73\u53f0\u53d1\u653e 2\uff1a\u5e97\u94fa\u9886\u53d6',\n  `status` tinyint(1) DEFAULT NULL COMMENT '\u72b6\u6001 0\uff1a\u672a\u4f7f\u7528 1\uff1a\u9501\u5b9a 2\uff1a\u5df2\u4f7f\u7528 3\uff1a\u5df2\u8fc7\u671f 4\uff1a\u5df2\u64a4\u56de',\n  `create_time` datetime DEFAULT NULL COMMENT '\u521b\u5efa\u65f6\u95f4',\n  `update_time` datetime DEFAULT NULL COMMENT '\u4fee\u6539\u65f6\u95f4',\n  `del_flag` tinyint(1) DEFAULT NULL COMMENT '\u5220\u9664\u6807\u8bc6 0\uff1a\u672a\u5220\u9664 1\uff1a\u5df2\u5220\u9664',\n  PRIMARY KEY (`id`),\n  UNIQUE KEY `idx_user_id_coupon_template_receive_count` (`user_id`,`coupon_template_id`,`receive_count`) USING BTREE,\n  KEY `idx_user_id` (`user_id`) USING BTREE\n) ENGINE=InnoDB AUTO_INCREMENT=1815640588360376337 DEFAULT CHARSET=utf8mb4 COMMENT='\u7528\u6237\u4f18\u60e0\u5238\u8868';<\/code><\/pre>\n\n\n\n<h2 class=\"wp-block-heading\">\u4f18\u60e0\u5238\u5206\u53d1\u903b\u8f91<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u9a8c\u8bc1\u524d\u7f6e\u903b\u8f91<\/h3>\n\n\n\n<p><strong>\u4e0d\u7ba1\u5728 service \u6216\u8005 controller\uff0c\u518d\u6216\u8005\u8bf4\u6211\u4eec\u7684\u6d88\u606f\u76d1\u542c\u5668\uff0c\u4e00\u5b9a\u8981\u5728\u524d\u9762\u53bb\u52a0\u4e0a\u5bf9\u5e94\u7684\u65e5\u5fd7\u3002\u56e0\u4e3a\u8fd9\u4e2a\u65e5\u5fd7\u6700\u91cd\u8981\u7684\u662f\u4ec0\u4e48\uff1f\u544a\u8bc9\u4f60\u8fd9\u4e2a\u6d88\u606f\u6709\u6ca1\u6709\u6267\u884c\uff0c\u4ee5\u53ca\u5b83\u7684\u6d88\u606f\u63d0\u793a\u662f\u4ec0\u4e48\u3002\u56e0\u4e3a\u5f88\u591a\u65f6\u5019\u6211\u4eec\u5728\u6392\u67e5\u7684\u65f6\u5019\uff0c\u5982\u679c\u4f60\u4e0d\u6253\u5370\u8fd9\u4e2a\u7684\u8bdd\uff0c\u603b\u6709\u4e00\u4e9b\u5947\u5947\u602a\u602a\u7684 bug\uff0c\u4f60\u6ca1\u529e\u6cd5\u5f88\u597d\u5730\u5b9a\u4f4d\u3002<\/strong><\/p>\n\n\n\n<p>\u5728\u4e4b\u524d\u7684\u7ae0\u8282\u4e2d\uff0c\u6211\u4eec\u4ec5\u521b\u5efa\u4e86\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u7684\u6d88\u606f\u751f\u4ea7\u73af\u8282\uff0c\u5c1a\u672a\u8fdb\u884c\u5b9e\u9645\u6d88\u8d39\u3002\u5728\u5206\u53d1\u670d\u52a1\u4e2d\uff0c\u6211\u4eec\u5c06\u7f16\u5199\u5177\u4f53\u7684\u6d88\u606f\u961f\u5217\u6d88\u8d39\u8005\uff0c\u5176\u4ee3\u7801\u903b\u8f91\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\n@Component\n@RequiredArgsConstructor\n@RocketMQMessageListener(\n        topic = \"one-coupon_distribution-service_coupon-task-execute_topic${unique-name:}\",\n        consumerGroup = \"one-coupon_distribution-service_coupon-task-execute_cg${unique-name:}\"\n)\n@Slf4j(topic = \"CouponTaskExecuteConsumer\")\npublic class CouponTaskExecuteConsumer implements RocketMQListener&lt;MessageWrapper&lt;CouponTaskExecuteEvent&gt;&gt; {\n\n    private final CouponTaskMapper couponTaskMapper;\n    private final CouponTemplateMapper couponTemplateMapper;\n    private final StringRedisTemplate stringRedisTemplate;\n    private final UserCouponMapper userCouponMapper;\n\n    @Override\n    public void onMessage(MessageWrapper&lt;CouponTaskExecuteEvent&gt; messageWrapper) {\n        \/\/ \u5f00\u5934\u6253\u5370\u65e5\u5fd7\uff0c\u5e73\u5e38\u53ef Debug \u770b\u4efb\u52a1\u53c2\u6570\uff0c\u7ebf\u4e0a\u53ef\u62a5\u5e73\u5b89\uff08\u6bd4\u5982\u6d88\u606f\u662f\u5426\u6d88\u8d39\uff0c\u91cd\u65b0\u6295\u9012\u65f6\u83b7\u53d6\u53c2\u6570\u7b49\uff09\n        log.info(\"&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u6b63\u5f0f\u6267\u884c - \u6267\u884c\u6d88\u8d39\u903b\u8f91\uff0c\u6d88\u606f\u4f53\uff1a{}\", JSON.toJSONString(messageWrapper));\n\n        \/\/ \u5224\u65ad\u4f18\u60e0\u5238\u6a21\u677f\u53d1\u9001\u72b6\u6001\u662f\u5426\u4e3a\u6267\u884c\u4e2d\uff0c\u5982\u679c\u4e0d\u662f\u6709\u53ef\u80fd\u662f\u88ab\u53d6\u6d88\u72b6\u6001\n        var couponTaskId = messageWrapper.getMessage().getCouponTaskId();\n        var couponTaskDO = couponTaskMapper.selectById(couponTaskId);\n        if (ObjectUtil.notEqual(couponTaskDO.getStatus(), CouponTaskStatusEnum.IN_PROGRESS.getStatus())) {\n            log.warn(\"&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u6b63\u5f0f\u6267\u884c - \u63a8\u9001\u4efb\u52a1\u8bb0\u5f55\u72b6\u6001\u5f02\u5e38\uff1a{}\uff0c\u5df2\u7ec8\u6b62\u63a8\u9001\", couponTaskDO.getStatus());\n            return;\n        }\n\n        \/\/ \u5224\u65ad\u4f18\u60e0\u5238\u72b6\u6001\u662f\u5426\u6b63\u786e\n        var queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                .eq(CouponTemplateDO::getId, couponTaskDO.getCouponTemplateId())\n                .eq(CouponTemplateDO::getShopNumber, couponTaskDO.getShopNumber());\n        var couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n        var status = couponTemplateDO.getStatus();\n        if (ObjectUtil.notEqual(status, CouponTemplateStatusEnum.ACTIVE.getStatus())) {\n            log.error(\"&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u6b63\u5f0f\u6267\u884c - \u4f18\u60e0\u5238ID\uff1a{}\uff0c\u4f18\u60e0\u5238\u6a21\u677f\u72b6\u6001\uff1a{}\", couponTaskDO.getCouponTemplateId(), status);\n            return;\n        }\n\n        \/\/ \u6b63\u5f0f\u5f00\u59cb\u6267\u884c\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\n        var readExcelDistributionListener = new ReadExcelDistributionListener(\n                couponTaskId,\n                couponTemplateDO,\n                stringRedisTemplate,\n                couponTemplateMapper,\n                userCouponMapper,\n                couponTaskMapper\n        );\n        EasyExcel.read(couponTaskDO.getFileAddress(), CouponTaskExcelObject.class, readExcelDistributionListener).sheet().doRead();\n    }\n}<\/code><\/pre>\n\n\n\n<p><code>var<\/code> \u5173\u952e\u5b57\u662f\u5728 <strong>JDK10<\/strong> \u4e2d\u5f15\u5165\u7684\u3002\u5b83\u7528\u4e8e\u5728 Java \u4e2d\u8fdb\u884c\u5c40\u90e8\u53d8\u91cf\u7c7b\u578b\u63a8\u65ad\uff0c\u65e8\u5728\u7b80\u5316\u4ee3\u7801\u7684\u7f16\u5199\uff0c\u540c\u65f6\u4fdd\u6301\u7c7b\u578b\u5b89\u5168\u6027\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>\u6211\u4e2a\u4eba\u4e0d\u662f\u5f88\u4e60\u60ef\u7528\uff0c\u6709\u70b9\u77ed\u8116\u5b50\u7684\u6837\u5f0f\uff0c\u5bf9\u4e8e\u6211\u8fd9\u79cd\u5f3a\u8feb\u75c7\u6765\u8bf4\uff0c\u6837\u5f0f\u975e\u5e38\u4e0d\u7f8e\u89c2\u3002\u8fd9\u91cc\u4ec5\u4ee3\u5165\u8ba9\u5927\u5bb6\u6709\u8fd9\u4e2a\u5173\u952e\u5b57\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-easy-excel\">2. EasyExcel \u8bfb\u53d6\u5206\u53d1\u7528\u6237\u5217\u8868<\/h3>\n\n\n\n<p>\u7531\u4e8e EasyExcel \u5e76\u975e\u8fd0\u884c\u5728 Spring \u73af\u5883\u4e2d\uff0c\u56e0\u6b64\u5f15\u5165 Spring Bean \u4f1a\u7a0d\u663e\u9ebb\u70e6\u3002\u5728\u6b64\uff0c\u6211\u4eec\u76f4\u63a5\u5728\u521b\u5efa\u6d88\u606f\u6d88\u8d39\u8005\u76d1\u542c\u7c7b\u65f6\uff0c\u901a\u8fc7\u6784\u9020\u51fd\u6570\u6ce8\u5165\u6240\u9700\u7684 Spring Bean\u3002<\/p>\n\n\n\n<p>\u4e3a\u4ec0\u4e48\u9a6c\u54e5\u4f60\u4e0d\u628a\u8fd9\u4e9b\u4e1c\u897f\u5728\u8fd9\u91cc\u9762\u76f4\u63a5\u901a\u8fc7 @Autowired \u6216\u8005\u8bf4 Spring \u7684\u65b9\u5f0f\u7ed9\u5b83\u6784\u9020\u8fdb\u6765\uff1f\u56e0\u4e3a\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\uff0c\u8fd9\u662f\u4e00\u4e2a\u5e38\u89c4\u7684\u7c7b\uff0c\u4e0d\u662f\u88ab Spring \u8fdb\u884c\u7ba1\u7406\u7684\u3002\u5927\u5bb6\u8fd9\u6837\u80fd\u7406\u89e3\u5417\uff1f\u76f8\u5f53\u4e8e\u6211\u4eec\u6bcf\u4e00\u6b21\u7684\u5206\u53d1\u7684\u4e00\u4e2a\u6d88\u8d39\uff0c\u90fd\u4f1a\u521b\u5efa\u4e00\u4e2a\u8fd9\u6837\u7684\u7c7b\u3002\u8fd9\u6837\u7684\u8bdd\uff0c\u6211\u4eec\u5c31\u6ca1\u529e\u6cd5\u901a\u8fc7\u6ce8\u5165\u7684\u65b9\u5f0f\u5c31\u8fc7\u53bb\u4e86\u3002\u6240\u4ee5\u8bf4\uff0c\u6211\u4eec\u7684\u6d88\u8d39\u8005\u4ed6\u5728\u8fd9\u91cc\u9762\u6709\u8fd9\u4e9b\u7c7b\u7684\u5f15\u7528\uff0c\u6211\u4eec\u76f4\u63a5\u7ed9\u4ed6\u4f20\u9012\u5c31\u597d\u4e86\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u4e0b\u9762\u8fd9\u4e2a\u662f\u6bd4\u8f83\u5e38\u89c4\u7684\u89e3\u6cd5\uff0c\u5c31\u662f\u901a\u8fc7\u5b83\u7684\u8bfb\u53d6\u7684\u65b9\u5f0f\uff0c\u628a\u5b83\u7684\u8fd9\u4e9b\u7c7b\u7ed9\u5b83\u653e\u8fdb\u6765\uff0c\u5b83\u5c31\u4f1a\u901a\u8fc7\u9ed8\u8ba4\u7684\u65b9\u5f0f\u8fdb\u884c\u4e00\u4e2a\u8bfb\u53d6\uff0c\u7136\u540e\u6211\u4eec\u76f4\u63a5\u7ed9\u5b83\u8df3\u8fc7\u6765\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u8fd9\u91cc\u9762\u7684\u8bdd\uff0c\u4ee3\u7801\u5176\u5b9e\u90fd\u8fd8\u6bd4\u8f83\u7b80\u5355\u3002\u9996\u5148\u7b2c\u4e00\u6b65\uff0c\u5c31\u662f\u6211\u4eec\u6784\u9020\u51fa\u6765\u5bf9\u5e94\u8be5\u7528\u6237\u7684 key\uff0c\u7136\u540e\u53bb\u7ed9\u5b83\u8fdb\u884c\u4e00\u4e2a\u81ea\u68c0\u3002\u5bf9\u5427\uff1f\u867d\u7136\u8fd9\u4e2a\u53eb\u201c\u81ea\u589e\u201d\uff0c\u4f46\u662f\u6211\u4eec\u7684\u503c\u662f -1\uff0c\u4e5f\u5c31\u662f\u81ea\u5df1\u3002\u7136\u540e\u5982\u679c\u8bf4\u6a21\u677f\u6263\u51cf\u5931\u8d25\u7684\u8bdd\uff0c\u4ed6\u5c31\u76f4\u63a5\u8fd4\u56de\u5417\uff1f\u6211\u4eec\u8fd9\u8fb9\u6ca1\u6709\u62a5\u9519\u7684\u8bdd\uff0c\u5176\u5b9e\u4e5f\u6ca1\u6709\u5f88\u5927\u7684\u5fc5\u8981\u6027\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u7684\u8bdd\uff0c\u4ed6\u4f1a\u53bb\u8fd9\u91cc\u9762\u8fdb\u884c\u4e00\u4e2a\u8c03\u7528\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u68c0\u67e5\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u3002\u7136\u540e\u5927\u5bb6\u8fd9\u4e00\u884c\u53ef\u4ee5\u5ffd\u7565<br>&lt;!--\u901a\u8fc7MySQL\u60b2\u89c2\u884c\u8bb0\u5f55\u9501\u786e\u4fdd\u5e93\u5b58\u4e0d\u4f1a\u88ab\u591a\u6263\uff0c\u5e76\u91c7\u7528\u4e0a\u5c42\u4e1a\u52a1\u81ea\u65cb\u91cd\u8bd5\u76f4\u81f3\u6210\u529f\u5c06\u4f18\u60e0\u5238\u5e93\u5b58\u6263\u51cf\u81f3\u96f6      \u5728 V2\u3001V3 \u7684\u65f6\u5019\u624d\u4f1a\u7528\u5230\u7684<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\n@RequiredArgsConstructor\npublic class ReadExcelDistributionListener extends AnalysisEventListener&lt;CouponTaskExcelObject&gt; {\n\n    private final Long couponTaskId;\n    private final CouponTemplateDO couponTemplateDO;\n    private final StringRedisTemplate stringRedisTemplate;\n    private final CouponTemplateMapper couponTemplateMapper;\n    private final UserCouponMapper userCouponMapper;\n    private final CouponTaskMapper couponTaskMapper;\n\n    @Override\n    public void invoke(CouponTaskExcelObject data, AnalysisContext context) {\n        \/\/ \u901a\u8fc7\u7f13\u5b58\u5224\u65ad\u4f18\u60e0\u5238\u6a21\u677f\u8bb0\u5f55\u5e93\u5b58\u662f\u5426\u5145\u8db3\n        String couponTemplateKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, couponTemplateDO.getId());\n        Long stock = stringRedisTemplate.opsForHash().increment(couponTemplateKey, \"stock\", -1);\n        if (stock &lt; 0) {\n            \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u5e93\u5b58\u4e0d\u8db3\u6263\u51cf\u5931\u8d25\n            return;\n        }\n\n        \/\/ \u6263\u51cf\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\uff0c\u5982\u679c\u6263\u51cf\u6210\u529f\uff0c\u8fd9\u91cc\u4f1a\u8fd4\u56de 1\uff0c\u4ee3\u8868\u4fee\u6539\u8bb0\u5f55\u6210\u529f\uff1b\u5426\u5219\u8fd4\u56de 0\uff0c\u4ee3\u8868\u6ca1\u6709\u4fee\u6539\u6210\u529f\n        int decrementResult = couponTemplateMapper.decrementCouponTemplateStock(couponTemplateDO.getShopNumber(), couponTemplateDO.getId(), 1);\n        if (!SqlHelper.retBool(decrementResult)) {\n            \/\/ \u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u5e93\u5e93\u5b58\u4e0d\u8db3\u6263\u51cf\u5931\u8d25\n            return;\n        }\n\n        \/\/ \u6dfb\u52a0\u7528\u6237\u9886\u5238\u8bb0\u5f55\u5230\u6570\u636e\u5e93\n        Date now = new Date();\n        DateTime validEndTime = DateUtil.offsetHour(now, JSON.parseObject(couponTemplateDO.getConsumeRule()).getInteger(\"validityPeriod\"));\n        UserCouponDO userCouponDO = UserCouponDO.builder()\n                .couponTemplateId(couponTemplateDO.getId())\n                .userId(Long.parseLong(data.getUserId()))\n                .receiveTime(now)\n                .receiveCount(1) \/\/ \u4ee3\u8868\u7b2c\u4e00\u6b21\u9886\u53d6\u8be5\u4f18\u60e0\u5238\n                .validStartTime(now)\n                .validEndTime(validEndTime)\n                .source(CouponSourceEnum.PLATFORM.getType())\n                .status(CouponStatusEnum.EFFECTIVE.getType())\n                .createTime(new Date())\n                .updateTime(new Date())\n                .delFlag(0)\n                .build();\n        try {\n            userCouponMapper.insert(userCouponDO);\n        } catch (BatchExecutorException bee) {\n            \/\/ \u7528\u6237\u5df2\u9886\u53d6\u4f18\u60e0\u5238\uff0c\u4f1a\u88ab\u552f\u4e00\u7d22\u5f15\u6821\u9a8c\u4f4f\uff0c\u76f4\u63a5\u8fd4\u56de\u5373\u53ef\n            return;\n        }\n\n        \/\/ \u6dfb\u52a0\u4f18\u60e0\u5238\u5230\u7528\u6237\u5df2\u9886\u53d6\u7684 Redis \u4f18\u60e0\u5238\u5217\u8868\u4e2d\n        String userCouponListCacheKey = String.format(EngineRedisConstant.USER_COUPON_TEMPLATE_LIST_KEY, data.getUserId());\n        String userCouponItemCacheKey = StrUtil.builder()\n                .append(couponTemplateDO.getId())\n                .append(\"_\")\n                .append(userCouponDO.getId())\n                .toString();\n        stringRedisTemplate.opsForZSet().add(userCouponListCacheKey, userCouponItemCacheKey, now.getTime());\n    }\n\n    @Override\n    public void doAfterAllAnalysed(AnalysisContext analysisContext) {\n        \/\/ \u786e\u4fdd\u6240\u6709\u7528\u6237\u90fd\u5df2\u7ecf\u63a5\u5230\u4f18\u60e0\u5238\u540e\uff0c\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u5b8c\u6210\u65f6\u95f4\n        CouponTaskDO couponTaskDO = CouponTaskDO.builder()\n                .id(couponTaskId)\n                .status(CouponTaskStatusEnum.SUCCESS.getStatus())\n                .completionTime(new Date())\n                .build();\n        couponTaskMapper.updateById(couponTaskDO);\n    }\n}\n<\/code><\/pre>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>Q\uff1a\u4e3a\u4ec0\u4e48\u6dfb\u52a0\u7528\u6237\u5df2\u9886\u53d6\u4f18\u60e0\u5238\u5217\u8868\u91cc\uff0c\u8981\u8ffd\u52a0\u7528\u6237\u9886\u5238\u8bb0\u5f55\u7684 ID \u5462\uff1f\u4e3a\u4ec0\u4e48\u4e0d\u662f\u5355\u72ec\u7684\u4f18\u60e0\u5238\u6a21\u677f ID\uff1f<\/em><\/p>\n\n\n\n<p><em>A\uff1a\u56e0\u4e3a\u4e00\u4e2a\u4f18\u60e0\u5238\u53ef\u80fd\u88ab\u7528\u6237\u9886\u53d6\u591a\u6b21\uff0c\u5982\u679c\u8bf4\u53ea\u4fdd\u5b58\u4f18\u60e0\u5238\u6a21\u677f ID\uff0c\u5c31\u6ca1\u6709\u529e\u6cd5\u6ee1\u8db3\u8fd9\u4e2a\u9700\u6c42\uff0c\u56e0\u4e3a ZSet \u4f1a\u8fc7\u6ee4\u91cd\u590d\u8bb0\u5f55\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<p>\u901a\u8fc7 MySQL \u60b2\u89c2\u884c\u8bb0\u5f55\u9501\u786e\u4fdd\u5e93\u5b58\u4e0d\u4f1a\u88ab\u591a\u6263\uff0cSQL \u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>&lt;!-- \u901a\u8fc7 MySQL \u60b2\u89c2\u884c\u8bb0\u5f55\u9501\u786e\u4fdd\u5e93\u5b58\u4e0d\u4f1a\u88ab\u591a\u6263\uff0c\u5e76\u91c7\u7528\u4e0a\u5c42\u4e1a\u52a1\u81ea\u65cb\u91cd\u8bd5\u76f4\u81f3\u6210\u529f\u5c06\u4f18\u60e0\u5238\u5e93\u5b58\u6263\u51cf\u81f3\u96f6 --&gt;\n&lt;!-- \u8fd9\u91cc\u8fd9\u884c\u6ce8\u91ca\u540e\u534a\u6bb5\u6709\u70b9\u95ee\u9898\uff0c\u4e0a\u5c42\u4e1a\u52a1\u91cd\u8bd5\u903b\u8f91\u662f v2 \u5206\u652f\u903b\u8f91 --&gt;\n&lt;update id=\"decrementCouponTemplateStock\"&gt;\n    UPDATE t_coupon_template\n    SET stock = stock - #{decrementStock}\n    WHERE shop_number = #{shopNumber}\n      AND id = #{couponTemplateId}\n      AND stock &gt;= #{decrementStock}\n&lt;\/update&gt;<\/code><\/pre>\n\n\n\n<p>\u8fd9\u91cc\u6709\u4e9b\u540c\u5b66\u53ef\u80fd\u4f1a\u6709\u7591\u95ee\uff0c\u4e3a\u4ec0\u4e48\u8981\u5148\u8bbf\u95ee\u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u8fdb\u884c\u6263\u51cf\uff0c\u7136\u540e\u518d\u6263\u51cf\u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u5e93\uff1f<\/p>\n\n\n\n<p>\u8fd9\u662f\u56e0\u4e3a\u5728\u540e\u7eed\u7684\u4f18\u60e0\u5238\u79d2\u6740\u6d41\u7a0b\u4e2d\uff0c\u6211\u4eec\u4e5f\u662f\u5148\u901a\u8fc7\u7f13\u5b58\u6263\u51cf\uff0c\u518d\u6263\u51cf\u6570\u636e\u5e93\u3002\u5982\u679c\u8fd9\u91cc\u5148\u6263\u51cf\u6570\u636e\u5e93\uff0c\u53ef\u80fd\u4f1a\u51fa\u73b0\u7f13\u5b58\u6263\u51cf\u6210\u529f\u4f46\u6570\u636e\u5e93\u6263\u51cf\u5931\u8d25\u7684\u60c5\u51b5\u3002\u4e3a\u4fdd\u6301\u4e00\u81f4\u6027\uff0c\u6211\u4eec\u5728\u6b64\u6d41\u7a0b\u4e2d\u540c\u6837\u5148\u6263\u51cf\u7f13\u5b58\uff0c\u7136\u540e\u518d\u6263\u51cf\u6570\u636e\u5e93\u3002<\/p>\n\n\n\n<p>\u6b64\u5916\uff0c\u5982\u679c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u8fc7\u8be5\u4f18\u60e0\u5238\uff0c\u4e3a\u4e86\u7b80\u5316\u6d41\u7a0b\u5904\u7406\uff0c\u6211\u4eec\u901a\u8fc7\u6355\u83b7\u552f\u4e00\u7d22\u5f15\u5f02\u5e38\u6765\u8df3\u8fc7\u5f53\u524d\u6d41\u7a0b\u3002<\/p>\n\n\n\n<p>\u6700\u540e\uff0c\u6211\u4eec\u4f7f\u7528 Redis \u7684 ZSet \u6765\u7f13\u5b58\u7528\u6237\u7684\u9886\u5238\u8bb0\u5f55\uff0c\u5e76\u5c06\u9886\u5238\u65f6\u95f4\u4f5c\u4e3a Score \u503c\uff0c\u8fd9\u6837\u7528\u6237\u5728\u67e5\u8be2\u65f6\u53ef\u4ee5\u6309\u65f6\u95f4\u5012\u5e8f\u663e\u793a\u9886\u53d6\u8bb0\u5f55\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u7136\u540e\u5728\u8fd9\u91cc\u9762\u53ef\u4ee5\u770b\u5230\uff0c\u6211\u4eec\u662f\u901a\u8fc7\u60b2\u89c2\u884c\u9501\u7684\u65b9\u5f0f\u6765\u786e\u4fdd\u5b83\u7684\u5e93\u5b58\u4e0d\u4f1a\u88ab\u591a\u6263\u3002\u4e4b\u524d\u5927\u5bb6\u5982\u679c\u770b\u8fc7\u6211\u4eec\u7684\u89c6\u9891\u7684\u8bdd\uff0c\u5e94\u8be5\u662f\u80fd\u591f\u7406\u89e3\u7684\u3002\u90a3\u5c31\u662f\u6211\u4eec\u901a\u8fc7 ID \u7684\u8fd9\u79cd\u65b9\u5f0f\uff0c\u7136\u540e\u53bb\u786e\u4fdd\u6211\u4eec\u901a\u8fc7\u60b2\u89c2\u9501\u7684\u5f62\u5f0f\uff0c\u7136\u540e\u53bb\u5bf9\u5b83\u8fdb\u884c\u4e00\u4e2a\u9501\u5b9a\uff0c\u7136\u540e\u4fdd\u969c\u6211\u4eec\u4e0d\u4f1a\u88ab\u591a\u6263\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u56e0\u4e3a\u6211\u4eec\u8fd9\u91cc\u9762\u52a0\u4e86\u4e00\u4e2a\u6761\u4ef6\uff1a\u5e93\u5b58\u5927\u4e8e\u7b49\u4e8e\u5b83\u5bf9\u5e94\u8981\u6263\u51cf\u7684\u4e00\u4e2a\u5e93\u5b58\uff0c\u6240\u4ee5\u8bf4\u8fd9\u91cc\u9762\u8fd8\u6709\u4e00\u70b9\u4e50\u89c2\u7684\u4e00\u4e9b\u673a\u5236\u3002<\/p>\n\n\n\n<p>\u6240\u4ee5\u8bf4\u5c31\u6709\u540c\u5b66\u53bb\u7ea0\u7ed3\uff1a\u201c\u9a6c\u54e5\uff0c\u6211\u5230\u5e95\u662f\u4e50\u89c2\u8fd8\u662f\u60b2\u89c2\uff1f\u201d\u8bf4\u5b9e\u5728\u8bdd\uff0c\u4e4b\u524d\u5927\u5bb6\u5728\u7f51\u4e0a\u5b66\u79d2\u6740\uff0c\u5bf9\u5427\uff1f\u5927\u5bb6\u4eba\u4e91\u4ea6\u4e91\u8fd9\u4e2a\u201c\u4e50\u89c2\u201d\u3002\u4f46\u662f\u5728\u6211\u770b\u6765\u7684\u8bdd\uff0c\u5305\u62ec\u6211\u4e4b\u524d\u4e5f\u662f\u8fd9\u4e48\u8ba4\u4e3a\u7684\uff0c\u540e\u9762\u5176\u5b9e\u6211\u53c8\u4ed4\u7ec6\u601d\u8003\u4e00\u4e0b\u8fd9\u4ef6\u4e8b\u60c5\u3002\u5728\u6211\u76ee\u524d\u770b\u6765\uff0c\u5176\u5b9e\u8fd9\u4e2a\u672c\u8d28\u4e0a\u662f\u60b2\u89c2\u7684\u3002\u56e0\u4e3a\u4e50\u89c2\u9501\u7684\u4e00\u4e2a\u5bd3\u610f\uff0c\u5c31\u662f\u5728\u4e0d\u4f7f\u7528\u9501\u7684\u60c5\u51b5\u4e0b\uff0c\u5bf9\u8fd9\u4e2a\u8bb0\u5f55\u8fdb\u884c\u5e76\u53d1\u5b89\u5168\u7684\u8bbf\u95ee\uff0c\u5bf9\u5427\uff1f\u4f46\u662f\u6211\u4eec\u8fd9\u91cc\u9762\u4e70\u7684\u8d2d\u7684\u88ab\u52a0\u4e86\u884c\u9501\uff0c\u5b83\u5176\u5b9e\u5df2\u7ecf\u662f\u4e00\u79cd\u6392\u4ed6\u9501\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u6240\u4ee5\u8bf4\u6211\u4e0d\u89c9\u5f97\u5b83\u662f\u4e50\u89c2\u9501\u3002<\/p>\n\n\n\n<p>\u5f53\u7136\uff0c\u4e4b\u524d\u661f\u7403\u91cc\u9762\u4e5f\u6709\u5f88\u591a\u5c0f\u4f19\u4f34\u633a\u7ea0\u7ed3\u7684\u3002\u6709\u4e9b\u9762\u8bd5\u5b98\u8bf4\u662f\u4e50\u89c2\uff0c\u90a3\u5c31\u662f\u4e50\u89c2\uff0c\u5bf9\u5427\uff1f\u4e0d\u8981\u5728\u9762\u8bd5\u7684\u65f6\u5019\u8ddf\u9762\u8bd5\u5b98\u53bb\u62ac\u8fd9\u79cd\u6760\u3002<\/p>\n\n\n\n<p>\u5728\u8fd9\u91cc\u9762\u6211\u4eec\u7ee7\u7eed\u8fdb\u884c\u3002\u7136\u540e\u7b49\u6211\u4eec\u9886\u5b8c\u4e4b\u540e\uff0c\u6211\u4eec\u76f4\u63a5\u53bb\u628a\u8fd9\u4e2a\u8bb0\u5f55\u7ed9\u653e\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u4e00\u4e2a\u6570\u636e\u5e93\u91cc\u9762\u3002\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\uff0c\u6211\u8fd8\u662f\u86ee\u559c\u6b22\u4f7f\u7528\u8fd9\u79cd\u6784\u5efa\u8005\u6a21\u5f0f\u53bb\u8fdb\u884c\u521b\u5efa\u5bf9\u8c61\u7684\u2014\u2014\u8fd9\u79cd\u4e00\u4e2a\u662f\u7f8e\u89c2\uff0c\u4e00\u4e2a\u662f\u5bf9\u7f8e\u89c2\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u8fd9\u91cc\u9762\u6211\u4eec\u901a\u8fc7\u4e86\u4e00\u4e2a\u5b83\u7684\u4f2a\u7d22\u5f15\u3001\u5b83\u7684\u4e00\u4e2a\u5f02\u5e38\u53bb\u8fdb\u884c\u4e00\u4e2a\u8865\u8d27\u3002\u56e0\u4e3a\u6211\u4eec\u521a\u624d\u6709\u52a0\u4f2a\u7d22\u5f15\uff0c\u5982\u679c\u4ed6\u5df2\u7ecf\u9886\u53d6\u8fc7\u4e86\uff0cOK\uff0c\u6211\u4eec\u76f4\u63a5\u6821\u9a8c\u7ec4\u8fd4\u56de\u5373\u53ef\u3002\u6211\u4eec\u628a\u8fd9\u4e2a\u8bb0\u5f55\u52a0\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u4e00\u4e2a\u7f13\u5b58\u91cc\u9762\uff0c\u8fd9\u6837\u7684\u8bdd\u5c31\u80fd\u591f\u89e3\u51b3\u95ee\u9898\u4e86\u3002<\/p>\n<\/blockquote>\n\n\n\n<p><\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6267\u884c\u4f18\u60e0\u5238\u5206\u53d1<\/h2>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u7136\u540e\u8fd9\u91cc\u9762\u6211\u4eec\u901a\u8fc7\u4e86\u4e00\u4e2a\u5b83\u7684\u4f2a\u7d22\u5f15\u3001\u5b83\u7684\u4e00\u4e2a\u5f02\u5e38\u53bb\u8fdb\u884c\u4e00\u4e2a\u8865\u8d27\u3002\u56e0\u4e3a\u6211\u4eec\u521a\u624d\u6709\u52a0\u4f2a\u7d22\u5f15\uff0c\u5982\u679c\u4ed6\u5df2\u7ecf\u9886\u53d6\u8fc7\u4e86\uff0cOK\uff0c\u6211\u4eec\u76f4\u63a5\u6821\u9a8c\u7ec4\u8fd4\u56de\u5373\u53ef\u3002\u6211\u4eec\u628a\u8fd9\u4e2a\u8bb0\u5f55\u52a0\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u4e00\u4e2a\u7f13\u5b58\u91cc\u9762\uff0c\u8fd9\u6837\u7684\u8bdd\u5c31\u80fd\u591f\u89e3\u51b3\u95ee\u9898\u4e86\u3002<\/p>\n\n\n\n<p>\u518d\u7136\u540e\u7684\u8bdd\uff0c\u6211\u4eec\u5728\u6700\u540e\u4e00\u884c\u2014\u2014\u8fd9\u4e2a\u662f Excel \u5b83\u5bf9\u5e94\u7684\u4e00\u4e2a API\u3002\u5982\u679c\u662f\u6700\u540e\u4e00\u884c\u7684\u8bdd\uff0c\u6267\u884c\u5b8c\u4e4b\u540e\uff0c\u6211\u4eec\u4f1a\u628a\u2026\u2026\u7528\u6237\u4ed6\u4e0d\u5bf9\uff0c\u4e0d\u662f\u7528\u6237\uff0c\u6211\u4eec\u4f1a\u628a\u4f18\u60e0\u5238\u7684\u63a8\u9001\u7684\u8fd9\u6761\u8bb0\u5f55\u7ed9\u4ed6\u8bbe\u7f6e\u4e00\u4e2a\u5b8c\u6210\u65f6\u95f4\u548c\u4e00\u4e2a\u72b6\u6001\u3002<\/p>\n\n\n\n<p>\u6211\u4eec\u8fd9\u91cc\u9762\u2026\u2026\u4e0d\u597d\u610f\u601d\uff0c\u6211\u4eec\u8fd9\u91cc\u9762\u76f4\u63a5\u628a\u65ad\u70b9\u653e\u5f00\uff0c\u7136\u540e\u6211\u4eec\u53bb\u770b\u5b83\u8fd9\u91cc\u9762\u5728\u8dd1\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u53bb\u770b\u4e00\u4e0b\u8fd9\u6761\u8bb0\u5f55\uff0c\u770b\u5230\u6ca1\u6709\uff1f\u6211\u4eec\u7684\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u4e00\u76f4\u5728\u5f80\u4e0b\u964d\u3002\u7136\u540e\u6211\u4eec\u53ef\u4ee5\u770b\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u5206\u53d1\u4efb\u52a1\uff0c\u5bf9\u5427\uff1f\u5b83\u7684\u72b6\u6001\u662f 1\u30021 \u7684\u8bdd\u662f\u4ec0\u4e48\uff1f\u5927\u5bb6\u770b\u4e00\u4e0b\uff0c1 \u7684\u8bdd\u662f\u201c\u6267\u884c\u4e2d\u201d\u3002<\/p>\n\n\n\n<p>\u6211\u4eec\u53ef\u4ee5\u53bb\u7b49\u4e00\u4e0b\u4ed6\u3002\u7b49\u5230\u5e93\u5b58\u53d8\u6210 1 \u7684\u65f6\u5019\uff0c\u6211\u4eec\u7684\u8fd9\u6761\u5206\u53d1\u4efb\u52a1\u5176\u5b9e\u5c31\u5df2\u7ecf\u7ed3\u675f\u4e86\u3002\u53ef\u80fd\u8981\u7a0d\u7b49\u4e00\u5c0f\u4f1a\u513f\uff0c\u56e0\u4e3a\u6211\u4eec V1 \u6700\u5927\u7684\u95ee\u9898\u5c31\u662f\uff1a\u5b83\u7684\u6240\u6709\u7684\u8bb0\u5f55\u90fd\u662f\u5355\u6b21\u548c\u6570\u636e\u5e93\u8fdb\u884c\u4ea4\u4e92\u7684\u3002\u8fd9\u6837\u7684\u8bdd\uff0c\u5176\u5b9e\u5b83\u7684\u6027\u80fd\u5168\u90e8\u90fd\u5361\u5728\u7f51\u7edc IO \u4e0a\u9762\u4e86\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u5c31\u662f\u6211\u4eec\u5728\u8ba1\u7b97\u673a\u7684\u4e16\u754c\u91cc\u9762\uff0cCPU\u2014\u2014\u7eaf CPU\u2014\u2014\u5df2\u7ecf\u662f\u975e\u5e38\u5feb\u7684\uff0c\u4f46\u662f\u95ee\u9898\u5361\u5728\u548c\u78c1\u76d8 IO \u548c\u7f51\u7edc IO \u7684\u4e00\u4e2a\u6d88\u8017\u4e0a\u9762\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>OK\uff0c\u6211\u4eec\u53ef\u4ee5\u770b\u5230\u8fd9\u4e2a\u5df2\u7ecf\u7ed3\u675f\u4e86\u3002\u5305\u62ec\u6211\u4eec V2\u3001V3 \u7248\u672c\uff0c\u5176\u5b9e\u66f4\u591a\u7684\u5c31\u662f\u901a\u8fc7\u6279\u91cf\u7684\u8fd9\u79cd\u601d\u60f3\u53bb\u89e3\u51b3\u8fd9\u4e2a\u95ee\u9898\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u53bb\u770b\u4e00\u4e0b\uff0c\u53ef\u4ee5\u770b\u5230\u6211\u4eec\u7684\u8fd9\u4e2a\u72b6\u6001\u5df2\u7ecf\u53d8\u6210\u4e86\u2026\u2026\u4e0d\u5bf9\uff0c\u6211\u4eec\u7684\u5e93\u5b58\u5df2\u7ecf\u53d8\u6210 1\uff0c\u7136\u540e\u6211\u4eec\u7684\u72b6\u6001\u5df2\u7ecf\u53d8\u6210\u4e86 3\uff0c\u5e76\u4e14\u6211\u4eec\u5b8c\u6210\u65f6\u95f4\u5df2\u7ecf\u7ed3\u675f\u4e86\u3002<\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u521b\u5efa\u4f18\u60e0\u5238\u6a21\u677f<\/h3>\n\n\n\n<p>\u521b\u5efa\u65b0\u7684\u4f18\u60e0\u5238\u6a21\u677f\uff0cRocketMQ 5.x \u9ed8\u8ba4\u4efb\u610f\u5ef6\u8fdf\u6d88\u606f\u6700\u5927\u95f4\u9694 3 \u5929\uff0c\u6240\u4ee5\u5927\u5bb6\u81ea\u5df1\u542f\u52a8 RocketMQ 5.x \u7684\u8bdd\uff0c\u8bb0\u5f97\u8bbe\u7f6e\u7ed3\u675f\u65f6\u95f4\u5728\u8ddd\u79bb\u5f53\u524d\u65f6\u95f4 3 \u5929\u5185\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>{\n  \"name\": \"\u7528\u6237\u4e0b\u5355\u6ee110\u51cf3\u7279\u5927\u4f18\u60e0\",\n  \"source\": 0,\n  \"target\": 1,\n  \"goods\": \"\",\n  \"type\": 0,\n  \"validStartTime\": \"2024-07-08 12:00:00\",\n  \"validEndTime\": \"2024-08-30 23:59:59\",\n  \"stock\": 20000,\n  \"receiveRule\": \"{\\\"limitPerPerson\\\":10,\\\"usageInstructions\\\":\\\"3\\\"}\",\n  \"consumeRule\": \"{\\\"termsOfUse\\\":10,\\\"maximumDiscountAmount\\\":3,\\\"explanationOfUnmetConditions\\\":\\\"3\\\",\\\"validityPeriod\\\":\\\"48\\\"}\"\n}<\/code><\/pre>\n\n\n\n<p>\u56e0\u4e3a\u661f\u7403\u901a\u7528\u4e91\u670d\u52a1\u5668\u5df2\u7ecf\u53d8\u66f4\u4e86\u8fd9\u4e2a\u6700\u5927\u5ef6\u65f6\u65f6\u95f4\u914d\u7f6e\u4e3a\u4e00\u5e74\uff0c\u6240\u4ee5\u8fd9\u91cc\u65f6\u95f4\u957f\u4e00\u70b9\u65e0\u6240\u8c13\u3002\u4fee\u6539 RocketMQ 5.x \u5ef6\u8fdf\u65f6\u95f4\u914d\u7f6e\u8be6\u60c5\u89c1\u4e0b\u6587\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-excel\">2. \u521b\u5efa Excel \u5206\u53d1\u8bb0\u5f55<\/h3>\n\n\n\n<p>\u6211\u4eec\u4e4b\u524d\u521b\u5efa\u7684 <code>ExcelGenerateTests#testExcelGenerate<\/code> \u5355\u5143\u6d4b\u8bd5\u9ed8\u8ba4\u521b\u5efa 100 \u4e07\u7684\u5206\u53d1\u8bb0\u5f55\uff0c\u5176\u5b9e\u662f\u6709\u70b9\u5927\u54c8\uff0c\u54b1\u4eec\u81ea\u5df1\u6267\u884c\u7684\u8bdd\uff0c\u53ea\u9700\u8981\u5c11\u91cf\u7684\u6570\u636e\u5373\u53ef\uff0c\u8981\u4e0d\u7136\u6267\u884c\u8d77\u6765\u592a\u6162\u4e5f\u4e0d\u592a\u884c\u3002<\/p>\n\n\n\n<p>\u8fd9\u91cc\u6211\u5efa\u8bae 5001\uff0c\u4e3a\u4ec0\u4e48\u6709\u96f6\u6709\u6574\u7684\uff1f\u56e0\u4e3a\u4e0b\u4e00\u8282\u4f1a\u7528\u5230\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>public final class ExcelGenerateTests {\n\u200b\n    \/**\n     * \u5199\u5165\u4f18\u60e0\u5238\u63a8\u9001\u793a\u4f8b Excel \u7684\u6570\u636e\uff0c\u81ea\u884c\u63a7\u5236\u5373\u53ef\n     *\/\n    private final int writeNum = 5001;\n    private final Faker faker = new Faker(Locale.CHINA);\n    private final String excelPath = Paths.get(\"\").toAbsolutePath().getParent() + \"\/tmp\";\n  \n    \/\/ ......\n}<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u521b\u5efa\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1<\/h3>\n\n\n\n<p>\u5927\u5bb6\u66ff\u6362\u81ea\u5df1\u672c\u5730\u7684 <code>fileAddress<\/code> \u5730\u5740\uff0c\u5e76\u4e14\u590d\u5236\u521a\u624d\u7684\u4f18\u60e0\u5238\u6a21\u677f ID\uff0c<code>sendType<\/code> \u4e3a\u7acb\u5373\u53d1\u9001\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>{\n  \"taskName\": \"\u53d1\u9001\u767e\u4e07\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\",\n  \"fileAddress\": \"\/Users\/machen\/workspace\/opensource\/onecoupon-rebuild\/tmp\/oneCoupon\u4efb\u52a1\u63a8\u9001Excel.xlsx\",\n  \"notifyType\": \"0,3\",\n  \"couponTemplateId\": 1829344152127647745,\n  \"sendType\": 0\n}<\/code><\/pre>\n\n\n\n<p>\u63d0\u4ea4\u540e\u5c31\u4f1a\u770b\u5230\u6211\u4eec\u7684\u5206\u53d1\u670d\u52a1\u5f00\u59cb\u6267\u884c\u4e86\uff0c\u6d88\u8d39\u8005\u65e5\u5fd7\u6253\u5370\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>2024-08-30T10:07:11.860+08:00INFO74088---&#91;g-mading0924_17]CouponTaskExecuteConsumer:&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u6b63\u5f0f\u6267\u884c - \u6267\u884c\u6d88\u8d39\u903b\u8f91\uff0c\u6d88\u606f\u4f53\uff1a{\"keys\":\"1829340057182523394\",\"message\":{\"couponTaskId\":1829340057182523394},\"timestamp\":1724983631789,\"uuid\":\"b56eb4dd-6d9e-48a6-b42c-c31068559419\"}<\/code><\/pre>\n\n\n\n<p>\u5927\u5bb6\u53ef\u4ee5\u8bb0\u4e00\u4e0b\u81ea\u5df1\u7535\u8111\u8dd1\u8fd9\u4e9b\u8bb0\u5f55\u9700\u8981\u591a\u957f\u65f6\u95f4\uff0c\u7136\u540e\u4e0b\u4e00\u8282\u6211\u4eec\u505a\u91cd\u6784\u65f6\uff0c\u5c31\u4f1a\u9707\u60ca\u5bf9\u5e94\u7684\u6027\u80fd\u63d0\u5347\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"4\">4. \u67e5\u8be2\u6267\u884c\u7ed3\u679c<\/h3>\n\n\n\n<p>\u901a\u8fc7\u67e5\u8be2\u6570\u636e\u5e93\u5f97\u77e5\uff0c\u6211\u4eec\u7684\u6570\u636e\u5747\u5300\u6563\u843d\u5728\u4e86 2 \u4e2a\u5e93\u7684 32 \u5f20\u8868\u91cc\uff0c\u521a\u597d 5000 \u6761\uff08\u6211\u6d4b\u8bd5\u7684\u65f6\u5019\u7528\u4e86 5000 \u6761\u8bb0\u5f55\uff09\u7528\u6237\u9886\u5238\u8bb0\u5f55\u3002<\/p>\n\n\n\n<p>\u88ab\u6807\u8bb0\u7684\u5c31\u662f\u6211\u4eec\u7684\u6570\u636e\u5e93\u8868\u91cc\u6709\u591a\u5c11\u6570\u636e\u884c\u6570\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image\"><img decoding=\"async\" src=\"https:\/\/oss.open8gu.com\/image-20240830103233713.png\" alt=\"\" title=\"\"\/><\/figure>\n\n\n\n<figure class=\"wp-block-image\"><img decoding=\"async\" src=\"https:\/\/oss.open8gu.com\/image-20240830103209554.png\" alt=\"\" title=\"\"\/><\/figure>\n\n\n\n<p>\u6211\u7528\u672c\u5730\u8dd1\u7684 RocketMQ \u6267\u884c 5000 \u6761\u7528\u6237\u4f18\u60e0\u5238\u5206\u53d1\u8bb0\u5f55\uff0c\u5dee\u4e0d\u591a\u7528\u4e86 1\u5206\u949f\uff0c\u4e91\u670d\u52a1\u5668 RocketMQ \u6d4b\u8bd5\u4e86\u4e0b\u4f1a\u591a\u4e2a 10%-20% \u7684\u6d88\u8017\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u67e5\u770b\u4f18\u60e0\u5238\u6a21\u677f\u8868\u7684\u5e93\u5b58\u6263\u51cf\uff0c\u521a\u597d\u662f\u548c Excel \u4e2d\u8bb0\u5f55\u7684\u6570\u636e\u4fdd\u6301\u4e00\u81f4\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image\"><img decoding=\"async\" src=\"https:\/\/oss.open8gu.com\/image-20240830102647870.png\" alt=\"\" title=\"\"\/><\/figure>\n\n\n\n<h2 class=\"wp-block-heading\">\u5e38\u89c1\u95ee\u9898\u7b54\u7591<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1-rocket-mq\">1. RocketMQ \u5ef6\u65f6\u65f6\u95f4<\/h3>\n\n\n\n<p>\u9ed8\u8ba4 RocketMQ \u6700\u5927\u5ef6\u65f6\u65f6\u95f4 3 \u5929\uff0c\u5982\u679c\u6211\u4eec\u53d1\u9001\u5ef6\u65f6\u6267\u884c\u65f6\u95f4\u8d85\u8fc7\u8fd9\u4e2a\u6570\u503c\uff0c\u5219\u4f1a\u62a5\u4ee5\u4e0b\u9519\u8bef\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>org.springframework.messaging.MessagingException: CODE: 13  DESC: timer message illegal, the delay time should not be bigger than the max delay 259200000ms; or if set del msg, the delay time should be bigger than the current time BROKER: xxx.xxx.xxx.xxx:10911\n<\/code><\/pre>\n\n\n\n<p>\u9519\u8bef\u6765\u6e90\u662f\u4ece Broker \u629b\u51fa\u6765\u7684\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>case WHEEL_TIMER_MSG_ILLEGAL:\n    response.setCode(ResponseCode.MESSAGE_ILLEGAL);\n    response.setRemark(String.format(\"timer message illegal, the delay time should not be bigger than the max delay %dms; or if set del msg, the delay time should be bigger than the current time\",\n        this.brokerController.getMessageStoreConfig().getTimerMaxDelaySec() * 1000L));<\/code><\/pre>\n\n\n\n<p><code>getTimerMaxDelaySec<\/code> \u65b9\u6cd5\u8fd4\u56de\u7684\u662f\u9ed8\u8ba4\u503c\uff0c\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>private int timerMaxDelaySec = 3600 * 24 * 3;\n\u200b\npublic int getTimerMaxDelaySec() {\n    return timerMaxDelaySec;\n}<\/code><\/pre>\n\n\n\n<p>\u56e0\u4e3a\u6211\u4eec\u516c\u5171\u4e91\u670d\u52a1\u5668\u5df2\u7ecf\u4fee\u6539\u4e86\u8fd9\u4e2a\u914d\u7f6e\uff0c\u6240\u4ee5\u5927\u5bb6\u5c31\u7b97\u8d85\u8fc7 3 \u5929\u4e5f\u65e0\u6240\u8c13\u7684\u3002\u90a3\u600e\u4e48\u53d8\u66f4\u8fd9\u4e2a\u5ef6\u65f6\u65f6\u95f4\u5462\uff1f<\/p>\n\n\n\n<p>\u5176\u5b9e\u5f88\u7b80\u5355\uff0c\u53ea\u9700\u8981\u5728 RocketMQ \u7684 Broker.conf \u6587\u4ef6\u4e2d\u52a0\u4e00\u884c\u53c2\u6570\u5373\u53ef\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>timerMaxDelaySec = 31622400<\/code><\/pre>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>timerMaxDelaySec \u7684\u6570\u636e\u5355\u4f4d\u9ed8\u8ba4\u4e3a\u79d2\uff0c\u6240\u4ee5\u6211\u4eec\u91c7\u53d6\u95f0\u5e74\u7684\u4e00\u5e74\u79d2\u6570\u5373\u53ef\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<p>\u8fd9\u6837\u7684\u8bdd\uff0c\u6211\u4eec\u5728\u4f18\u60e0\u5238\u6a21\u677f\u521b\u5efa\u7684\u65f6\u5019\uff0c\u8fd8\u9700\u8981\u5728\u4f18\u60e0\u5238\u6a21\u677f\u7684\u6570\u636e\u6821\u9a8c\u4e2d\u52a0\u5165\u6709\u6548\u671f\u4e0d\u80fd\u8d85\u8fc7\u5f53\u524d\u65f6\u95f4\u4e00\u5e74\u7684\u9a8c\u8bc1\u903b\u8f91\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-rocket-mq\">2. RocketMQ \u8d85\u957f\u5ef6\u65f6\u65f6\u95f4\u662f\u5426\u4f1a\u88ab\u6e05\u7406\uff1f<\/h3>\n\n\n\n<p>\u4e4b\u524d\u6211\u6709\u70b9\u62c5\u5fc3\u8bbe\u7f6e\u8fd9\u4e48\u957f\u65f6\u95f4\u7684\u5ef6\u65f6\u65f6\u95f4\uff0c\u548c RocketMQ \u7684\u6d88\u606f\u6e05\u7406\u7b56\u7565\u662f\u5426\u4f1a\u6709\u51b2\u7a81\uff1f<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p><em>RocketMQ \u6d88\u606f\u6e05\u7406\u7b56\u7565\u5373\u6d88\u606f\u53d1\u9001\u540e\uff0c\u5728\u6307\u5b9a\u65f6\u95f4\u540e\u4f1a\u88ab\u5220\u9664\uff0c\u817e\u51fa\u78c1\u76d8\u7a7a\u95f4\uff0c\u9ed8\u8ba4\u662f 48 \u5c0f\u65f6\u540e\u6e05\u7406\u3002<\/em><\/p>\n<\/blockquote>\n\n\n\n<p>\u7ecf\u8fc7\u6211\u5b9e\u9645\u6d4b\u8bd5\uff0c\u628a\u6e05\u7406\u65f6\u95f4\u6539\u4e3a 1 \u5c0f\u65f6\uff0c\u7136\u540e\u5ef6\u65f6\u65f6\u95f4\u6539\u4e3a 2 \u5c0f\u65f6\uff0c\u5ef6\u65f6\u6d88\u606f\u53ef\u4ee5\u6b63\u5e38\u6267\u884c\u3002\u6211\u4e2a\u4eba\u731c\u6d4b\uff0c\u5ef6\u65f6\u6d88\u606f\u7684\u961f\u5217\u662f\u4e2a\u7279\u6b8a\u961f\u5217\uff0c\u5e76\u4e0d\u5c5e\u4e8e\u666e\u901a\u88ab\u6e05\u7406\u7684\u8303\u7574\u3002\u540e\u9762\u5982\u679c\u6709\u673a\u4f1a\u770b\u770b RocketMQ \u7684\u6e90\u7801\uff0c\u8fd9\u91cc\u518d\u7ed9\u5927\u5bb6\u8865\u5145\u8be6\u7ec6\u5185\u5bb9\u3002<\/p>\n\n\n\n<p>\u5b8c\u7ed3\uff0c\u6492\u82b1 \ud83c\udf89<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5728\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u91cc\u9762\uff0c\u5176\u5b9e\u4e0a\u9762\u6709\u8bb2\u5230\uff0c\u5c31\u662f\u8bf4\u5b83\u662f\u652f\u6301\u4e00\u4e9b\u4efb\u610f\u7684\u5ef6\u8fdf\uff0c\u6bd4\u5982\u8bf4\u201c\u6211\u4e09\u5929\u540e\u6267\u884c\u201d\uff0c\u6bd4\u5982\u8bf4\u201c\u6211\u660e\u5929\u4e0b\u5348 5 \u70b9\u6267\u884c\u201d\u3002\u5728\u4e4b\u524d\u6211\u4eec RocketMQ 5.0 \u7248\u672c\u4e4b\u524d\u662f\u5b9e\u73b0\u4e0d\u4e86\u7684\uff0c\u56e0\u4e3a\u5b83\u4e4b\u524d\u7684\u5185\u5bb9\u662f\u901a\u8fc7\u53eb\u505a\u56fa\u5b9a\u7684\u4e00\u4e2a\u5ef6\u65f6\u673a\u5236\u3002\u4f46\u662f 5.0 \u4e4b\u540e\uff0c\u5b83\u5176\u5b9e\u4f18\u5316\u4e86\u4e00\u4e2a\u4e8b\u60c5\uff1a\u901a\u8fc7\u65f6\u95f4\u8f6e\u7684\u7b97\u6cd5\u53bb\u89e3\u51b3\u8fd9\u4e2a\u95ee\u9898\u3002<\/p>\n\n\n\n<p>\u4f46\u662f\u603b\u4e4b\uff0c\u54b1\u4eec\u5c31\u5b98\u65b9\u7684\u89d2\u5ea6\u4e0a\u6765\u8003\u8651\uff0c\u5b83\u7ed9\u7684\u6700\u5927\u5ef6\u8fdf\u65f6\u95f4\u662f\u4e09\u5929\u3002\u5982\u679c\u8bf4\u6211\u4eec\u7684\u53d1\u9001\u65f6\u95f4\u8d85\u8fc7\u4e86\u8fd9\u4e2a\u503c\uff0c\u5c31\u4f1a\u62a5\u9519\u3002\u8fd9\u4e2a\u662f\u4ece\u535a\u5ba2\u90a3\u8fb9\u5f04\u8fc7\u6765\u7684\uff0c\u5f53\u65f6\u6211\u5c31\u770b\u4e86\u4e00\u4e0b\u5b83\u7684\u6e90\u7801\uff0c\u7136\u540e\u5c31\u5728\u2026\u2026\u5176\u5b9e\u5927\u5bb6\u53bb\u628a RocketMQ \u62c9\u4e0b\u6765\u4e00\u770b\u5c31\u77e5\u9053\u4e86\u3002<\/p>\n\n\n\n<p>\u56e0\u4e3a\u6211\u662f\u7528\u7684\u6211\u73b0\u5728\u7528\u7684\u672c\u5730\u7684\u6d88\u606f\u961f\u5217\uff0c\u6240\u4ee5\u8bf4\u6211\u4e5f\u662f\u5df2\u7ecf\u6539\u8fc7\u4e86\u3002\u5f53\u7136\u6211\u4eec\u516c\u5171\u7684\u4e91\u670d\u52a1\u5668\u5176\u5b9e\u4e5f\u5df2\u7ecf\u6539\u8fc7\u4e86\u3002\u5982\u679c\u8bf4\u5927\u5bb6\u60f3\u8981\u53bb\u6539\u7684\u8bdd\uff0c\u53ea\u9700\u8981\u5728\u8fd9\u4e2a RocketMQ \u7684\u914d\u7f6e\u6587\u4ef6\u91cc\u9762\u52a0\u4e00\u4e2a\u53c2\u6570\u5c31\u53ef\u4ee5\u4e86\u3002\u7136\u540e\u5b83\u7684\u9ed8\u8ba4\u5355\u4f4d\u662f\u79d2\uff0c\u7136\u540e\u6211\u4eec\u76f4\u63a5\u53d6\u4e00\u5e74\u7684\u65f6\u95f4\uff0c\u8fd9\u6837\u5c31\u53ef\u4ee5\u4e86\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u53ef\u80fd\u4f1a\u6709\u540c\u5b66\u4f1a\u7ea0\u7ed3\u8bf4\uff1a\u201c\u9a6c\u54e5\uff0c\u6211\u8fd9\u4e48\u957f\u65f6\u95f4\u7684\u5ef6\u8fdf\u65f6\u95f4\uff0c\u5bf9\u4e8e\u6d88\u606f\u961f\u5217\u6765\u8bf4\u662f\u5426\u4f1a\u6709\u538b\u529b\uff1f\u201d<\/p>\n\n\n\n<p>\u6211\u53ef\u4ee5\u5f88\u76f4\u767d\u5730\u8bb2\uff1a\u5c31\u662f\u4e0d\u4f1a\u6709\u538b\u529b\u3002\u4e3a\u4ec0\u4e48\uff1f\u9996\u5148\u4f60\u7684\u6570\u636e\u91cf\u4e0d\u4f1a\u5f88\u591a\u2014\u2014\u4f60\u60f3\uff0c\u8fd0\u8425\u4eba\u5458\u53bb\u521b\u5efa\u8fd9\u79cd\u5206\u53d1\u7684\u5ef6\u65f6\u4efb\u52a1\uff0c\u5b83\u4e0d\u4f1a\u6709\u591a\u5c11\u3002\u5176\u6b21\uff0c\u5c31\u7b97\u6709\u5f88\u591a\uff0c\u4e00\u7ea7\u522b 10 \u4ebf\u7ea7\u522b\u2014\u2014\u633a\u591a\u4e86\u2014\u2014\u4f46\u662f\u4f60\u8981\u77e5\u9053\uff0c\u6211\u4eec\u5728\u6d88\u606f\u961f\u5217\u91cc\u9762\uff0c\u5b83\u7684\u4e00\u6761\u6d88\u606f\u662f\u4ee5\u78c1\u76d8\u7684\u5f62\u5f0f\u5b58\u50a8\u7684\uff0c10 \u4ebf\u6761\u6d88\u606f\u5b83\u7684\u91cf\u624d\u591a\u5927\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u6240\u4ee5\u8bf4\u6211\u89c9\u5f97\u4e0d\u4f1a\u6709\u95ee\u9898\u3002<\/p>\n\n\n\n<p>\u7136\u540e\uff0c\u56e0\u4e3a\u6211\u4eec RocketMQ \u9ed8\u8ba4\u7684\u8bdd\uff0c\u5b83\u662f\u6709\u4e00\u4e2a\u78c1\u76d8\u7684\u6e05\u7406\u7b56\u7565\u7684\u3002\u4e0d\u8fc7\u5b83\u7684\u78c1\u76d8\u6e05\u7406\u7b56\u7565\u662f\u9488\u5bf9\u90a3\u4e9b\u5df2\u7ecf\u6d88\u8d39\u8fc7\u7684\u6d88\u606f\u3002\u6240\u4ee5\u8bf4\uff0c\u6211\u4eec\u8fd9\u4e9b\u6ca1\u6709\u6d88\u8d39\u8fc7\u7684\u6d88\u606f\u4e5f\u4e0d\u4f1a\u88ab\u5904\u7406\u3002<\/p>\n<\/blockquote>\n\n\n\n<p><\/p>\n\n\n\n<p><\/p>\n\n\n\n<p><strong>zouhu\uff1a\u9a6c\u54e5\uff0c\u540e\u53f0\u7ba1\u7406\u670d\u52a1\u7684\u53d1\u9001\u5206\u53d1\u4efb\u52a1\u548c\u5206\u53d1\u670d\u52a1\u7684\u53d1\u9001\u4f18\u60e0\u52b5\u4e4b\u95f4\u7684\u5173\u7cfb\u6709\u70b9\u5f04\u4e0d\u61c2\uff1f\u53d1\u9001\u5206\u53d1\u4efb\u52a1\u4e3a\u4ec0\u4e48\u8981\u63d2\u5165\u5230\u6570\u636e\u5e93\uff0c\u5206\u53d1\u4efb\u52a1\u4e0d\u662f\u901a\u8fc7\u6d88\u606f\u961f\u5217\u6765\u8bfb\u53d6\u7684\u5417\uff1f<\/strong><\/p>\n\n\n\n<p><strong>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;zouhu\uff1a\u540e\u53f0\u7ba1\u7406\u8d1f\u8d23\u521b\u5efa\u3001\u67e5\u770b\u3001\u4fee\u6539\u5404\u79cd\u8fd0\u8425\u6240\u9700\u8981\u7684\u8bb0\u5f55\u3002\u5206\u53d1\u670d\u52a1\u8d1f\u8d23\u5c06\u4f18\u60e0\u5238\u5206\u7ed9\u7528\u6237\u624b\u91cc\uff0c\u4e24\u4e2a\u804c\u8d23\u4e0d\u540c\u3002 \u63d2\u5165\u6570\u636e\u5e93\u662f\u56e0\u4e3a\u521b\u5efa\u8bb0\u5f55\u8981\u7559\u75d5\uff0c\u8981\u67e5\u770b\u8fdb\u5ea6\u7b49\u7b49\u3002<\/strong><\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p><strong>\u5e0c\u671b~\u65e0\u5c3d\uff1a\u9a6c\u54e5\uff0c\u600e\u4e48\u7406\u89e3\uff0c\u5982\u679c\u8fd9\u91cc\u5148\u6263\u51cf\u6570\u636e\u5e93\uff0c\u53ef\u80fd\u4f1a\u51fa\u73b0\u7f13\u5b58\u6263\u51cf\u6210\u529f\u4f46\u6570\u636e\u5e93\u6263\u51cf\u5931\u8d25\u7684\u60c5\u51b5\u3002\u6211\u6709\u70b9\u4e0d\u61c2\u4e3a\u4ec0\u4e48\u6570\u636e\u5e93\u4f1a\u6263\u51cf\u5931\u8d25\uff1f<\/strong><img loading=\"lazy\" decoding=\"async\" height=\"211\" width=\"1308\" src=\"https:\/\/i-blog.csdnimg.cn\/direct\/fb8a4f47febb4e61a6ce09e81a1e2f30.png\" alt=\"\"><\/p>\n\n\n\n<p><strong>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;\u5e0c\u671b~\u65e0\u5c3d\uff1a\u6bd4\u5982\u8bf4\u6570\u636e\u5e93\u6263\u51cf\u8d85\u65f6\uff0c\u518d\u6bd4\u5982\u8bf4\u6570\u636e\u5e93\u91cc\u9762\u7684\u5e93\u5b58\u4f59\u989d\u6ca1\u6709\u4e86\uff0c\u90fd\u6709\u53ef\u80fd\u7684<\/strong><\/p>\n\n\n\n<p><strong>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u5e0c\u671b~\u65e0\u5c3d\uff1aman\uff0c\u8981\u6ce8\u610f\u5728\u300a\u4f18\u60e0\u5238\u5206\u53d1\u6d41\u7a0b\u4e2d\u300b\u5148\u6263\u51cf\u7f13\u5b58\u5e93\u5b58\uff0c\u518d\u6263\u51cf\u6570\u636e\u5e93\u5e93\u5b58\uff0c\u662f\u4e3a\u4e86\u548c\u540e\u7eed\u300a\u4f18\u60e0\u5238\u79d2\u6740\u6d41\u7a0b\u300b\u5339\u914d\uff0c\u56e0\u4e3a\u300a\u4f18\u60e0\u5238\u79d2\u6740\u6d41\u7a0b\u300b\u5c31\u662f\u5148\u6263\u51cf\u7f13\u5b58\u5e93\u5b58\uff0c\u518d\u6263\u51cf\u6570\u636e\u5e93\u5e93\u5b58\u3002 \u6211\u4eec\u5148\u5047\u8bbe\u4f18\u60e0\u5238\u5e93\u5b58\u548c\u6570\u636e\u5e93\u5e93\u5b58\u90fd\u53ea\u52691\u4e2a\u5e93\u5b58\uff1a \u5047\u8bbe\u4f60\u5728\u300a\u4f18\u60e0\u5238\u5206\u53d1\u6d41\u7a0b\u4e2d\u300b\u5148A\uff1a\u6263\u51cf\u6570\u636e\u5e931\u4e2a\u5e93\u5b58\uff0c\u518dB\uff1a\u6263\u51cf\u7f13\u5b581\u4e2a\u5e93\u5b58\uff1b \u4f60\u5728\u300a\u4f18\u60e0\u5238\u79d2\u6740\u6d41\u7a0b\u4e2d\u300b\u5148B\uff1a\u6263\u51cf\u7f13\u5b581\u4e2a\u5e93\u5b58\uff0c\u518dA\uff1a\u6263\u51cf\u6570\u636e\u5e931\u4e2a\u5e93\u5b58\uff0c \u90a3\u4e48\u6709\u53ef\u80fd\u51fa\u73b0\u4ee5\u4e0b\u6d41\u7a0b\uff1a 1.\u5206\u53d1\u6d41\u7a0bA\u2014\u2014\u300b2.\u79d2\u6740\u6d41\u7a0bB\u2014\u2014\u300b3.\u79d2\u6740\u6d41\u7a0bA\u2014\u2014\u300b4.\u5206\u53d1\u6d41\u7a0bB \u5f53\u4f60\u6267\u884c\u5b8c1\u30012\u7684\u65f6\u5019\uff0c\u6b64\u65f6\u6570\u636e\u5e93\u5e93\u5b58\u548c\u7f13\u5b58\u5e93\u5b58\u90fd\u4e3a0\uff0c\u90a3\u4e48\u4f60\u518d\u6267\u884c3\u7684\u65f6\u5019\uff0c\u5c31\u4f1a\u5931\u8d25\u3002 \u4ee5\u4e0a\u662f\u6211\u81ea\u5df1\u7684\u7406\u89e3\uff0c\u5982\u6709\u4e0d\u5bf9\u8bf7\u6307\u6b63<\/strong><\/p>\n\n\n\n<p><strong>warrior&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u6709\u70b9\u4e0d\u7406\u89e3<\/strong><\/p>\n\n\n\n<p><strong>\u57fa\u6d1b\u592b&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u662f\u4e0d\u662f\u7c7b\u4f3c\u4e8e\u907f\u514d\u6b7b\u9501\u7684\u90a3\u79cd\u7ed9\u8d44\u6e90\u6392\u5e8f\u7684\u65b9\u6848<\/strong><\/p>\n\n\n\n<p><strong>\u57fa\u6d1b\u592b&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u5927\u4f6c \u8fd8\u6ca1\u5b66\u5230\u4f18\u60e0\u5238\u79d2\u6740\u90e8\u5206 \u8bf7\u95ee\u662f\u4e0d\u662f\u4ed6\u4eec\u53ef\u4ee5\u5171\u7528\u4e00\u5957\u4f18\u60e0\u5238\u6a21\u677f\uff1f \u6240\u4ee5\u8fd9\u91cc\u8981\u8003\u8651\u4e24\u4e2a\u4e1a\u52a1\u7684\u6263\u51cf\u5e93\u5b58\u7684\u987a\u5e8f\u95ee\u9898\uff1f<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u5176\u5b9e\u6211\u60f3\u95ee\uff0c\u540c\u4e00\u5f20\u4f18\u60e0\u5238\u771f\u7684\u4f1a\u6709\u540c\u65f6\u88ab\u5206\u53d1\u548c\u79d2\u6740\u7684\u65f6\u5019\u5417\uff0c\u88ab\u79d2\u6740\u7684\u4f18\u60e0\u5238\u5f80\u5f80\u5e94\u8be5\u5e93\u5b58\u4e0d\u591a\uff0c\u5206\u53d1\u7684\u4f18\u60e0\u5238\u5e94\u8be5\u5f80\u5f80\u5e93\u5b58\u5341\u5206\u5145\u8db3\u5427<\/strong><\/p>\n\n\n\n<p><strong>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4e00\u5207\u7686\u6709\u53ef\u80fd\uff0c\u5b9e\u73b0\u4e1a\u52a1\u8981\u8003\u8651\u4efb\u4f55\u60c5\u51b5<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u4f46\u5b9e\u9645\u4e0a\u5f88\u591a\u65f6\u5019\u4e5f\u4e0d\u4f1a\u8003\u8651\u6570\u636e\u5e93\u5b95\u673a\u7b49\u6982\u7387\u5f88\u4f4e\u7684\u60c5\u51b5\u5440<\/strong><\/p>\n\n\n\n<p><strong>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u522b\u729f<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u7167\u4f60\u8fd9\u4e48\u8bf4\u90a3\u6211\u53ef\u4e0d\u53ef\u4ee5\u628a\u4e00\u5207\u8131\u79bb\u4e1a\u52a1\u5b9e\u9645\u7684\u95ee\u9898\u90fd\u8bf4\u6210\u662f\u729f\u5462\ud83d\ude28<\/strong><\/p>\n\n\n\n<p><strong>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u53c8\u729f<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u62bd\u8c61\u4e86\u54c8\u8001\u54e5\uff0c\u4e0d\u662f\u4f60\u80fd\u56de\u7b54\u7684\u5c31\u662f\u201c\u6240\u6709\u60c5\u51b5\u90fd\u8981\u8003\u8651\u201d\uff0c\u4e0d\u80fd\u56de\u7b54\u7684\u5c31\u662f\u201c\u729f\u201d<\/strong><\/p>\n\n\n\n<p><strong>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6211\u7761\u89c9\u4e86\uff0c\u660e\u5929\u8ba9\u6211\u597d\u597d\u60f3\u60f3\u600e\u4e48\u56de\u7b54\u4f60<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u4e2a\u4eba\u7406\u89e3\u662f\uff0c\u4e0d\u4f1a\u5bf9\u6b63\u5e38\u4e1a\u52a1\u903b\u8f91\u9020\u6210\u5f88\u5927\u5f71\u54cd\u7684\u95ee\u9898\uff0c\u987a\u5e26\u4e5f\u5c31\u89e3\u51b3\u4e86\uff0c\u6210\u672c\u5f88\u4f4e\u3002\u5176\u4ed6\u6709\u4e9b\u95ee\u9898\u867d\u7136\u4e5f\u4f1a\u53d1\u751f\uff0c\u4f46\u53ef\u80fd\u8ba9\u4ee3\u7801\u53d8\u590d\u6742\uff0c\u5c31\u8003\u8651\u89e3\u51b3\u6210\u672c<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6211\u89c9\u5f97\u8fd9\u79cd\u5e93\u5b58\u4e0d\u8db3\u7684\u573a\u666f\u591a\u5c11\u6709\u70b9\u626f\uff0c\u8fd9\u79cd\u6279\u91cf\u53d1\u7684\u90fd\u662f\u5ec9\u4ef7\u5238\uff0c\u79d2\u6740\u7684\u5238\u90fd\u4f1a\u5355\u72ec\u641e\uff0c\u751a\u81f3\u4f1a\u63a5\u53e3\u9694\u79bb\uff0c\u5982\u679c\u662f\u7ea2\u5305\u96e8\u7684\u8bdd\u8fd8\u4f1aredis\u5e93\u5b58\u5206\u7247<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u786e\u5b9e\u5f88\u626f\uff0c\u767e\u5206\u4e4b99.9999\u90fd\u4e0d\u4f1a\u53d1\u751f\u8fd9\u79cd\u95ee\u9898<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4e00\u822c\u90fd\u5b81\u53ef\u5e93\u5b58\u8d85\u4e86\u4e5f\u4e0d\u4f1a\u5c11\uff0c\u9664\u975e\u662f\u610f\u6599\u4e4b\u5916\u7684\u7206\u6b3e<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u5206\u53d1\u4f18\u60e0\u5238\u4e5f\u4e0d\u5b58\u5728\u7206\u6b3e\u4e0d\u7206\u6b3e\u4e86\uff0c\u90fd\u662f\u7531\u5546\u5bb6\u6765\u51b3\u5b9a\u60f3\u53d1\u591a\u5c11\u5c31\u53d1\u591a\u5c11\uff0c\u800c\u4e14\u4e00\u822c\u8fd9\u79cd\u4f18\u60e0\u5238\u4e5f\u4e0d\u80fd\u7531\u7528\u6237\u81ea\u5df1\u53bb\u9886\u53d6<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4f18\u60e0\u5238\u7684\u8bdd\uff0c\u6709\u4e00\u79cd\u662f\u4e8f\u6b20\u4e5f\u4f1a\u53d1\u7684\uff0c\u7eaf\u7eaf\u5c31\u662f\u4e3a\u4e86\u5f15\u6d41\uff0c\u8fd9\u79cd\u7684\u8bdd\u80af\u5b9a\u662f\u9650\u91cf\u7684\uff0c\u4f46\u662f\u8fd9\u4e2a\u5c5e\u4e8e\u662f\u79d2\u6740\u4e1a\u52a1\u5c42\u9762\u4e86\u3002\u800c\u8fd9\u79cd\u5927\u6279\u91cf\u53d1\u7684\u5238\u90fd\u662f\u5f88\u5ec9\u4ef7\u7684\u90a3\u79cd\u5c0f\u989d\u5238\uff0c\u7406\u8bba\u4e0a\u662f\u5e73\u53f0\u65b9\u4f1a\u53d1\uff0c\u5e73\u53f0\u65b9\u6709\u7528\u6237\u7684\u4e2a\u4eba\u4fe1\u606f\uff0c\u6574\u7406\u51faExcel\u7136\u540e\u5bf9Excel\u52a0\u5de5\u5904\u7406\uff0c\u6309\u7167sheet\uff0c\u7ed3\u5408\u7ebf\u7a0b\u6c60\u548cmq\u7684\u5e76\u53d1\u6d88\u8d39\u8fdb\u884c\u6d77\u91cf\u7684\u5206\u53d1<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u5224\u65ad\u662f\u5426\u7ed3\u675f\u4e5f\u662f\u6839\u636esheet\u8bfb\u53d6\u7684\u540e\u5904\u7406\u91cc\u9762\u7684\u903b\u8f91\u5b9e\u73b0\u7684\u3002\u4e5f\u6ca1\u5fc5\u8981\u53d1\u6d88\u606f\uff0c\u6700\u540e\u4e00\u6279\u6b21\u4e86\uff0c\u76f4\u63a5\u5728\u65b9\u6cd5\u91cc\u9762\u6267\u884c\uff0c\u8fd8\u80fd\u8ba1\u7b97\u4efb\u52a1\u5b8c\u6210\u4e2a\u6570\uff0c\u7528\u6765\u66f4\u65b0\u72b6\u6001<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u5238\u7684\u8bdd\uff0c\u5305\u62ec\u5e73\u53f0\u5238\u548c\u5546\u5bb6\u5238\uff0c\u8fd9\u79cd\u5ec9\u4ef7\u5238\u5927\u591a\u90fd\u662f\u5e73\u53f0\u5238<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6839\u636e\u521a\u624d\u7684\u903b\u8f91\uff0c\u6839\u672c\u4e0d\u53ef\u80fd\u8d85\u53d1\uff0c\u591a\u8bbe\u7f6e\u5e93\u5b58\u90fd\u884c<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u7528\u7ebf\u7a0b\u6c60\u7684\u8bdd\u5c31\u4e0d\u9700\u8981mq\u4e86\u5427<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u9700\u8981\u7684\uff0c\u6027\u80fd\u5dee\u4e86\u5f88\u591a\uff0c\u4e00\u4e2a\u7ebf\u7a0b\u5904\u7406\u4e00\u4e2asheet\uff0c\u4e00\u4e2asheet\u6279\u91cf\u53d1\uff0c\u6bcf\u4e2a\u6279\u6b21\u5e76\u53d1\u6d88\u8d39\uff0c\u6700\u540e\u4e00\u4e2a\u6279\u6b21\u5728\u540e\u5904\u7406\u6267\u884c\uff0c\u5982\u679c\u5b58\u6279\u91cf\u8bb0\u5f55\u7684list\u4e3a\u7a7a\u5c31\u76f4\u63a5\u66f4\u65b0\u8fdb\u5ea6\uff0c\u4efb\u52a1\u8fdb\u5ea6\u8ba1\u6570\uff0c\u4efb\u52a1\u72b6\u6001\u3002\u6bcf\u4e2a\u7ebf\u7a0b\u8d1f\u8d23\u4e00\u4e2asheet\uff0c\u4e5f\u5c31\u662f\u8bf4\u4f1a\u5728redis\u770b\u5230\u5341\u4e2a\u8fdb\u5ea6\u6761\u3002\u76ee\u524d\u8fd9\u91cc\u7684\u95ee\u9898\u662f\u7528\u4ec0\u4e48\u4f5c\u4e3a\u8fdb\u5ea6\u6761\u7684key\uff0c\u5982\u679c\u662f\u4efb\u52a1ID\u7684\u8bdd\uff0c\u53ea\u80fd\u901a\u8fc7MySQL\u552f\u4e00\u5065\u5224\u65ad\u662f\u4e0d\u662f\u91cd\u590d\u9886\u53d6\uff0c\u5982\u679c\u662f\u7528\u6a21\u677fID\u7684\u8bdd\u53ef\u4ee5\u5224\u65ad\uff0c\u4f46\u662f\u53ea\u80fd\u4e00\u6b21\u6027\u4f7f\u7528\u8fd9\u4e2a\u6a21\u677f\uff0c\u60f3\u518d\u641e\u4e00\u4e2a\u5c31\u5f97\u91cd\u65b0\u521b\u5efa\uff0c\u7531\u4e8e\u5e42\u7b49\u5224\u65ad\uff0c\u53ea\u80fd\u5728\u7b2c\u4e00\u6b21\u521b\u5efa\u6a21\u677f\u5f00\u59cb\u6267\u884c\u5f00\u59cb\u4e4b\u540e\u4e24\u4e09\u79d2\u4e4b\u540e\u518d\u521b\u5efa\u4e00\u4e2a\u4e00\u6a21\u4e00\u6837\u7684\u65b0\u6a21\u677f\uff0c\u6211\u89c9\u5f97\u8fd9\u4e2a\u7b56\u7565\u5c31\u884c\u4e86\uff0c\u56e0\u4e3a\u8fd9\u79cd\u78c1\u76d8\u7a7a\u95f4\u5f88\u5ec9\u4ef7\uff0c\u4e3a\u4e86\u7ed9\u6bcf\u4e2a\u4eba\u591a\u53d1\u51e0\u4e2a\u91cd\u590d\u521b\u5efa\u4e00\u4e24\u4e2a\u529f\u80fd\u6709\u6548\u671f\u5565\u7684\u90fd\u4e00\u6837\u7684\u5238\u4e5f\u6ca1\u4ec0\u4e48\u3002\u5e42\u7b49\u6ce8\u89e3\u4e3b\u8981\u662f\u9632\u4e0d\u60f3\u591a\u521b\u5efa\u5374\u591a\u521b\u5efa\u7684\u573a\u666f\u3002<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4f60\u53ef\u4ee5\u7406\u89e3\u4e3a\u5982\u679c\u4e00\u4e2asheet\u5341\u4e07\u4e2a\uff0c\u4e00\u6279\u6b21\u4e94\u5343\uff0c\u4f60\u7528\u5355\u72ec\u7684\u670d\u52a1\u5668\u90e8\u7f72mq\uff0cmq\u53ef\u4ee5\u5341\u4e2a\u7ebf\u7a0b\u5e76\u53d1\u6d88\u8d39\u8fd9\u4e2asheet\u7684\u5206\u53d1\u4efb\u52a1\uff0c\u5e94\u7528\u670d\u52a1\u5668\u5982\u679c\u662f\u5341\u4e2a\u7ebf\u7a0b\uff0c\u90a3\u4e48\u7528\u4e86mq\u4e4b\u540e\u5c31\u662f\u8fdc\u8d85\u4ec5\u4ec5\u5341\u4e2a\u7ebf\u7a0b\u7684\u7ebf\u7a0b\u6c60\u80fd\u8fbe\u5230\u7684\u6027\u80fd\u3002<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6211\u81ea\u5df1\u6d4b\u4e86\uff0c\u4e0d\u7528\u7684\u8bdd\u534a\u5c0f\u65f6\uff0c\u7528\u4e86\u4e4b\u540e\u5c31\u516d\u5206\u534a\u5230\u4e03\u5206\u949f<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u8fd9\u4e5f\u4ec5\u4ec5\u662f\u81ea\u5df1\u7684\u7b14\u8bb0\u672c\u80fd\u63d0\u5347\u7684\uff0c\u5982\u679c\u662f\u751f\u4ea7\u7684\u670d\u52a1\u5668\u5462\uff1f\u96c6\u7fa4\u90e8\u7f72\u7684mq\uff0c\u9ad8\u6027\u80fd\u7684\u670d\u52a1\u5668\uff0c\u5982\u679c\u518d\u7528\u4e0a\u865a\u62df\u7ebf\u7a0b\uff0c\u76f4\u63a5\u8d77\u98de\uff0c\u4f1a\u975e\u5e38\u5feb<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4e0d\u8fc7\u8fd9\u91cc\u6709\u4e2a\u5751\uff0c\u5c31\u662f\u4e00\u5b9a\u8981\u627e\u63d2\u5165\u8bb0\u5f55\u518d\u6263\u51cfMySQL\u5e93\u5b58\uff0c\u56e0\u4e3a\u56db\u4e2a\u884c\u4e3a\u90fd\u5728\u4e00\u4e2a\u4e8b\u52a1\uff0c\u5982\u679c\u5148\u9501\u64cd\u4f5c\u6263\u51cf\u5e93\u5b58\u7684\u8bdd\uff0c\u4e8b\u52a1\u8fd8\u6ca1\u63d0\u4ea4\uff0c\u540e\u9762\u63d2\u5165\u8fd8\u662f\u957f\u4e8b\u52a1\uff0c\u5c31\u5f88\u5bb9\u6613\u9020\u6210\u9501\u8d85\u65f6\uff0c\u7136\u540e\u5c11\u53d1\uff0c\u6f0f\u53d1\uff0c\u4e00\u822c\u662f\u51e0\u4e07\u4e2a\u751a\u81f3\u5341\u591a\u4e07\u4e2a\u6f0f\u53d1\uff0c\u6240\u4ee5\u8981\u4e48\u5148\u7528\u4e00\u4e2a\u4e8b\u52a1\u6216\u8005\u5faa\u73af\u6307\u6570\u9000\u907f\u786e\u4fdd\u4e00\u5b9a\u6210\u529f\uff0c\u7136\u540e\u518d\u6267\u884c\u63d2\u5165\u548credis\u7684\u4e8b\u52a1\uff0c\u8981\u4e48\u5c31\u662f\u6211\u521a\u624d\u8bf4\u7684\u5148\u63d2\u5165\u8bb0\u5f55\u518d\u76f4\u884c\u9501\u64cd\u4f5c\uff0c\u6700\u540e\u540c\u6b65redis\u5e93\u5b58\u548c\u8bb0\u5f55\u3002\u4e8b\u52a1\u6210\u529f\u4e0e\u5426\u90fd\u4f1a\u5728\u540e\u9762\u8bb0\u5f55\u8fdb\u5ea6<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u90a3\u5176\u5b9e\u53ea\u8981\u628a\u73b0\u5728\u7684\u7248\u672c\u6539\u6210\u4e00\u4e2a\u7ebf\u7a0b\u5904\u7406\u4e00\u4e2asheet\u5c31\u884c\u4e86\uff0c\u540e\u9762mq\u7684\u5177\u4f53\u5206\u53d1\u903b\u8f91\u90fd\u4e0d\u7528\u548b\u53d8<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u540e\u9762\u4e5f\u662f\u8981\u53d8\u7684\uff0c\u4ed6\u90a3\u4e2a\u4f1a\u53d1\u5f88\u591a\u65e0\u7528\u7684\u6d88\u606f\uff0c\u6211\u8fd9\u4e2a\u7248\u672c\u6bcf\u4e2a\u7ebf\u7a0b\u7684\u4efb\u52a1\u53ea\u9700\u8981\u53d1\u5355\u4e2asheet\u91cc\u9762\u7684\u6279\u6b21\u4e2a\u6d88\u606f\u6216\u8005\u6279\u6b21\u51cf\u4e00\u4e2a\u6d88\u606f<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u4ed6\u539f\u6765\u7684\u903b\u8f91\u662f\u6bcf\u884c\u90fd\u4f1a\u53d1\u5230\u4e0b\u6e38\u5224\u65ad\uff0c\u8fd9\u6837\u7684\u8bdd\u65e0\u7528\u7684\u6d88\u606f\u592a\u591a\u4e86<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u800c\u4e14\u884c\u53f7\u53ef\u4ee5\u901a\u8fc7easyExcel\u7684\u5206\u6790\u4e0a\u4e0b\u6587\u76f4\u63a5\u83b7\u53d6\uff0c\u6ca1\u5fc5\u8981\u8ba1\u7b97++<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u76f4\u63a5\u5728\u4e0a\u6e38\u52a0\u4e00\u4e2a\u53d6\u4f59\u5224\u65ad\u5c31\u4e0d\u4f1a\u591a\u53d1\u4e86<\/strong><\/p>\n\n\n\n<p><strong>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u8fd8\u6709\u4e00\u4e2a\u95ee\u9898\u662f\uff0csheet\u62c6\u5206\u662f\u5e94\u8be5\u5728\u63d0\u4ea4excel\u65f6\u5c31\u63d0\u4ea4\u62c6\u5206\u597d\u7684\uff0c\u8fd8\u662f\u76f4\u63a5\u5148\u8fdb\u884c\u4e00\u6b21\u89e3\u6790\u5b8c\u6210\u62c6\u5206\uff0c\u518d\u53bb\u63d0\u4ea4\u4efb\u52a1\u5230\u7ebf\u7a0b\u6c60\u5462<\/strong><\/p>\n\n\n\n<p><strong>int main()&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u5148\u81ea\u5df1\u7528\u7a0b\u5e8f\u62c6\uff0c\u7136\u540e\u518d\u8bf7\u6c42\u63a5\u53e3\uff0c\u8fd9\u4e2a\u4e0d\u9ebb\u70e6\uff0c\u95eegpt\u5c31\u884c\uff0c\u81ea\u5df1\u6d4b\u8bd5\u7684\u8bdd\u76f4\u63a5\u7528faker\u751f\u6210\u5c31\u884c<\/strong><\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u5730\u4fe1\u54e5\uff1a\u9a6c\u54e5\u4f60\u8fd9\u4e24\u53e5\u8bdd\u662f\u4e0d\u662f\u81ea\u76f8\u77db\u76fe\u5440\uff0c\u6211\u6709\u70b9\u770b\u4e0d\u61c2\u4e86= =<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"422\" width=\"1027\" src=\"https:\/\/i-blog.csdnimg.cn\/direct\/64533ac884434b8fad747f179b369452.png\" alt=\"\"><\/p>\n\n\n\n<p>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;\u5730\u4fe1\u54e5\uff1a\u6211\u4eec\u5728\u5206\u53d1\u7684\u65f6\u5019\uff0c\u903b\u8f91\u548c\u666e\u901a\u9886\u53d6\u662f\u6709\u70b9\u513f\u4e0d\u4e00\u6837\u7684\u3002\u5206\u53d1\u903b\u8f91\u4e2d\uff0c\u53ea\u5224\u65ad\u7528\u6237\u9886\u53d6\u8fc7\uff0c\u76f4\u63a5\u8ba9\u4ed6\u8df3\u8fc7\uff0c\u4e0d\u5b58\u5728\u9886\u53d6\u591a\u6b21\u7684\u884c\u4e3a<\/p>\n\n\n\n<p>\u5730\u4fe1\u54e5&nbsp;\u56de\u590d&nbsp;\u9a6c\u4e01\uff1a\u55f7\u55f7\uff0c\u6211\u8fd8\u4ee5\u4e3a\u5206\u53d1\u548c\u9886\u53d6\u662f\u4e00\u4e2a\u610f\u601d\u3002\u90a3\u6211\u61c2\u4e86<\/p>\n\n\n\n<p>mni&nbsp;\u56de\u590d&nbsp;\u9a6c\u4e01\uff1a\u9a6c\u54e5\uff0c\u5982\u679c\u540c\u4e00\u79cd\u4f18\u60e0\u5238 \u7528\u6237\u5df2\u7ecf\u9886\u53d6\u4e86\u591a\u5f20\uff08\u591a\u4e8e1\u5f20\uff09\uff0c\u518d\u6267\u884c\u5206\u53d1\u4e14excel\u8868\u4e2d\u5b58\u5728\u8be5\u7528\u6237\uff0c\u6b64\u65f6receive_count\u4e0d\u4e3a1\u6ca1\u6cd5\u901a\u8fc7\u552f\u4e00\u7d22\u5f15\u8df3\u8fc7\u4e86\u5427\u3002<\/p>\n\n\n\n<p>Young&nbsp;\u56de\u590d&nbsp;mni\uff1a\u5bf9\uff0c\u6240\u4ee5\u5728\u5206\u53d1\u524d\u8981\u67e5\u8be2\u7528\u6237\u7684\u9886\u53d6\u6b21\u6570<\/p>\n\n\n\n<p>Scofield.&nbsp;\u56de\u590d&nbsp;mni\uff1a\u786e\u5b9e\u5b58\u5728\u8fd9\u79cd\u95ee\u9898<\/p>\n\n\n\n<p>Scofield.&nbsp;\u56de\u590d&nbsp;\u9a6c\u4e01\uff1a\u9a6c\u54e5\uff0c\u8fd9\u91cc\u662f\u4e0d\u662f\u4e0d\u5e94\u8be5\u7528\u552f\u4e00\u7d22\u5f15\u6765\u907f\u514d\u53d1\u9001\u5df2\u7ecf\u9886\u53d6\u8fc7\u7684\u4f18\u60e0\u5238\uff0c\u8fd9\u6837\u53ef\u80fd\u4f1a\u5bfc\u81f4\u7528\u6237\u9886\u53d6\u8fc7\u4e00\u6b21\u5238\uff0c\u8fd9\u6b21\u53d1\u653e\u8df3\u8fc7\u4e86\uff0c\u5982\u679c\u7528\u6237\u9886\u53d6\u8fc7\u5927\u4e8e\u4e00\u6b21\uff0c\u90a3\u4e48\u8fd8\u4f1a\u53d1\u653e\u3002\u5e94\u8be5\u5148\u67e5\u8be2\u7528\u6237\u9886\u53d6\u8be5\u4f18\u60e0\u5238\u6b21\u6570\u662f\u5426\u4e3a0\uff0c\u4e3a0\u518d\u53d1\u653e\uff0c\u4e0d\u4e3a0\u5219\u8bf4\u660e\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u8fc7\u4e86\u4e0d\u7528\u518d\u53d1\u653e\u4e86\u3002<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;mni\uff1a\u6211\u7406\u89e3\u7684\u662f\u6bcf\u6b21\u9886\u53d6\u90fd\u4f1a\u6dfb\u52a0\u4e00\u6761\u9886\u53d6\u8bb0\u5f55\uff0c\u800c\u4e0d\u662f\u5728\u539f\u7eaa\u5f55\u4e0a\u4fee\u6539\uff0c\u7b2c\u4e00\u6b21receive_count\u4e3a1\uff0c\u7b2c\u4e8c\u6b21receive_count\u4e3a2\uff0c\u5982\u679c\u6709receive_count &gt; 1\u7684\u8bb0\u5f55\uff0c\u90a3\u4e48\u4e00\u5b9a\u6709receive_count = 1\u7684\u8bb0\u5f55<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;Scofield.\uff1a\u6211\u7406\u89e3\u7684\u662f\u6bcf\u6b21\u9886\u53d6\u90fd\u4f1a\u6dfb\u52a0\u4e00\u6761\u9886\u53d6\u8bb0\u5f55\uff0c\u800c\u4e0d\u662f\u5728\u539f\u7eaa\u5f55\u4e0a\u4fee\u6539\uff0c\u7b2c\u4e00\u6b21receive_count\u4e3a1\uff0c\u7b2c\u4e8c\u6b21receive_count\u4e3a2\uff0c\u5982\u679c\u6709receive_count &gt; 1\u7684\u8bb0\u5f55\uff0c\u90a3\u4e48\u4e00\u5b9a\u6709receive_count = 1\u7684\u8bb0\u5f55<\/p>\n\n\n\n<p>\u4e0d\u7231\u5b66\u4e60\u7684\u61d2\u732b&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6211\u73b0\u5728\u6709\u70b9\u7591\u95ee\uff0c\u8fd9\u91cc\u7684receive_count\u5177\u4f53\u5728\u4ee3\u7801\u54ea\u91cc\u8fdb\u884c\u4fee\u6539\u4e86\u5417\uff1f\u6211\u5e76\u6ca1\u6709\u770b\u5230\uff0c\u8001\u54e5\u53ef\u4ee5\u8bf4\u4e00\u4e0b\u5417\uff1f<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u4e0d\u7231\u5b66\u4e60\u7684\u61d2\u732b\uff1a\u8fd9\u4e2a\u5b57\u6bb5\u7684\u610f\u601d\u662f\u6807\u8bc6\u7b2c\u51e0\u6b21\u9886\u53d6\uff0c\u7528\u6237\u81ea\u5df1\u4e3b\u52a8\u9886\u53d6\u7684\u4e1a\u52a1\u5e94\u5f53\u662f\u6ca1\u505a\u7684\uff0c\u8fd9\u91cc\u53ea\u662f\u8bf4\u5047\u5982\u7528\u6237\u9886\u53d6\u8fc7\u7684\u8bdd\uff0c\u6bcf\u4e00\u6b21\u9886\u53d6\u90fd\u4f1a\u5bf9\u5e94\u4e00\u6761\u6570\u636e\uff0c\u6240\u4ee5\u9886\u53d6\u8fc7\u591a\u6b21\u7684\u8bdd\u5c31\u4e00\u5b9a\u9886\u53d6\u4e86\u7b2c\u4e00\u6b21\uff0c\u5c31\u4e00\u5b9a\u6709receive_count\u7b49\u4e8e1\u7684\u8bb0\u5f55<\/p>\n\n\n\n<p>\u82b1\u5f00\u5bcc\u8d35&nbsp;\u56de\u590d&nbsp;mni\uff1a\u6211\u7684\u7406\u89e3\u662f\uff0c\u9886\u53d6\u548c\u5206\u53d1\u90fd\u662f\u6dfb\u52a0receive_count\u4e3a1\u7684\u8bb0\u5f55(\u6211\u4eec\u8fd8\u6ca1\u6709\u5199\u9886\u53d6\u7684\u4ee3\u7801\uff0c\u73b0\u5728\u5206\u53d1\u7684\u903b\u8f91\u662f\u8fd9\u6837\u7684)\uff0c\u800c\u6211\u4eec\u521a\u597d\u8bbe\u7f6e\u4e86\u552f\u4e00\u7d22\u5f15\uff0c\u8fd9\u5c31\u53ef\u4ee5\u4fdd\u8bc1\u6bcf\u4e2a\u4f18\u60e0\u5238\u4e0d\u7ba1\u662f\u5982\u4f55\u83b7\u53d6\u7684\u90fd\u53ea\u80fd\u9886\u53d6\u4e00\u6b21\uff0c\u4e5f\u4fdd\u8bc1\u4e86\u5982\u679c\u9886\u53d6\u4e4b\u540e\uff0c\u5728\u540e\u7eed\u7684\u5206\u53d1\u8fc7\u7a0b\u4e2d\u4e5f\u53ef\u4ee5\u901a\u8fc7\u552f\u4e00\u7d22\u5f15\u4fdd\u8bc1\u4e0d\u518d\u53d1\u653e\u7ed9\u7528\u6237\u3002<\/p>\n\n\n\n<p>\u82b1\u5f00\u5bcc\u8d35&nbsp;\u56de\u590d&nbsp;\u9e45\u9e45\u9e45\uff1a\u6211\u7684\u7406\u89e3\u662f\uff0c\u9886\u53d6\u548c\u5206\u53d1\u90fd\u662f\u6dfb\u52a0receive_count\u4e3a1\u7684\u8bb0\u5f55(\u6211\u4eec\u8fd8\u6ca1\u6709\u5199\u9886\u53d6\u7684\u4ee3\u7801\uff0c\u73b0\u5728\u5206\u53d1\u7684\u903b\u8f91\u662f\u8fd9\u6837\u7684)\uff0c\u800c\u6211\u4eec\u521a\u597d\u8bbe\u7f6e\u4e86\u552f\u4e00\u7d22\u5f15\uff0c\u8fd9\u5c31\u53ef\u4ee5\u4fdd\u8bc1\u6bcf\u4e2a\u4f18\u60e0\u5238\u4e0d\u7ba1\u662f\u5982\u4f55\u83b7\u53d6\u7684\u90fd\u53ea\u80fd\u9886\u53d6\u4e00\u6b21\uff0c\u4e5f\u4fdd\u8bc1\u4e86\u5982\u679c\u9886\u53d6\u4e4b\u540e\uff0c\u5728\u540e\u7eed\u7684\u5206\u53d1\u8fc7\u7a0b\u4e2d\u4e5f\u53ef\u4ee5\u901a\u8fc7\u552f\u4e00\u7d22\u5f15\u4fdd\u8bc1\u4e0d\u518d\u53d1\u653e\u7ed9\u7528\u6237\u3002\u54b1\u4fe9\u610f\u601d\u4e00\u6837\u5417<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;\u82b1\u5f00\u5bcc\u8d35\uff1a\u5982\u679c\u5e73\u53f0\u5238\u65e2\u53ef\u4ee5\u5206\u53d1\u4e5f\u53ef\u4ee5\u81ea\u884c\u9886\u53d6\u7684\u8bdd\uff0c\u90a3\u4e48\u5728Redis\u5e94\u8be5\u4e5f\u8981\u6709\u4e00\u4e2a\u7528\u6237\u5df2\u9886\u53d6\u6b21\u6570\u7684\u7f13\u5b58\uff0c\u7136\u540e\u81ea\u884c\u9886\u53d6\u7684\u65f6\u5019receive_count\u5c31\u662f\u8fd9\u4e2a\u7f13\u5b58\u6570 + 1\u3002\u5206\u53d1\u7684\u8bdd\u90fd\u662freceive_count\u4e3a1\uff0c\u56e0\u4e3a\u672c\u6765\u5c31\u8ba9\u9886\u4e00\u6b21\uff0c\u4e0d\u8bba\u662f\u81ea\u5df1\u5151\u6362\u8fc7\u4e5f\u597d\uff0c\u5df2\u7ecf\u88ab\u5206\u53d1\u4e86\u4e5f\u597d\uff0c\u90fd\u662f\u53ef\u4ee5\u9650\u5236\u4f4f\u7684<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u6ca1\u770b\u61c2\u201d\u6263\u51cf\u5e93\u5b58\u903b\u8f91\u7684\u8bdd\u540c\u6837\u662f\u91c7\u7528\u4e50\u89c2\u9501\u673a\u5236\u4fdd\u969c\u201c\u8fd9\u53e5\u8bdd\uff0c\u8fd9\u4e2a\u4e50\u89c2\u9501\u673a\u5236\u662f\u6307 MySQL \u5e95\u5c42\u4f7f\u7528\u4e86\u4e50\u89c2\u9501\u673a\u5236\uff1f<\/p>\n\n\n\n<p>\u4e50\u89c2\u9501\u673a\u5236\u6307\u7684\u662f update ...... where stock &gt;= \"decement\"\uff0c\u5728\u4fee\u6539\u7684\u65f6\u5019\u624d\u5224\u65ad\u5e93\u5b58\u662f\u5426\u8db3\u591f\u3002\u5982\u679c\u662f\u5148select\u62ff\u5230\u5e93\u5b58\uff0c\u7136\u540e\u81ea\u5df1\u5224\u65ad\u662f\u5426\u8db3\u591f\uff0c\u8db3\u591f\u7684\u8bdd\u518dupdate\uff0c\u4f46\u662f\u8fd9\u4e24\u4e2a\u64cd\u4f5c\u4e0d\u662f\u539f\u5b50\u6027\u7684\uff0c\u4f1a\u51fa\u73b0 \u5e93\u5b58 &lt; 0 \u7684\u60c5\u51b5<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u5206\u4eab\u4e00\u4e0b\u81ea\u5df1\u634b\u7684\u601d\u8def \u9996\u5148\u770badmin\u540e\u7ba1\u4e2d\u7684\u6267\u884c\u6d41\u7a0b 1.CouponTemplateServiceImpl\u7c7b \u6267\u884ccreateCouponTemplate\u65b9\u6cd5\u521b\u5efa\u4e86\u4f18\u60e0\u5238\uff0c\u7136\u540e\u8ba9\u8be5\u4f18\u60e0\u5238\u4fe1\u606f\u5b58\u5165\u6570\u636e\u5e93 2.CouponTaskServiceImpl\u7c7b \u6267\u884ccreateCouponTask\u65b9\u6cd5\u521b\u5efa\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\uff0c\u7136\u540e\u8ba9\u8be5\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u5b58\u5165\u6570\u636e\u5e93 \u8fd9\u4e2a\u4efb\u52a1\u662f\u5206\u53d1\u6307\u5b9a\u7684\u4f18\u60e0\u5238\u7ed9Excel\u4e2d\u7684\u7528\u6237\uff0c\u7edf\u8ba1Excel\u4e2d\u7684\u884c\u6570\u662f\u901a\u8fc7\u7ebf\u7a0b\u6c60\u5f02\u6b65\u6267\u884c\u7684\u3002 \u5982\u679c\u5206\u53d1\u4efb\u52a1\u662f\u7acb\u5373\u53d1\u9001\uff0c\u90a3\u72b6\u6001\u5c31\u662f\u6267\u884c\u4e2d\uff0c\u5982\u679c\u4e0d\u662f\u7acb\u5373\u53d1\u9001\uff0c\u4e5f\u5c31\u662f\u5b9a\u65f6\u53d1\u9001\uff0c\u90a3\u72b6\u6001\u5c31\u662f\u5f85\u6267\u884c \u5982\u679c\u662f\u7acb\u5373\u53d1\u9001\uff0c\u5c31\u901a\u8fc7mq\u53d1\u9001\u8be5\u5206\u53d1\u4efb\u52a1id \u518d\u6765\u770bdistribution\u5206\u53d1\u7684\u6d41\u7a0b\uff0c\u6ce8\u610f\u6211\u4eec\u5206\u53d1\u7684\u65f6\u5019\uff0c\u662f\u5206\u53d1\u6307\u5b9a\u7684\u4f18\u60e0\u5238\u7ed9Excel\u4e2d\u7684\u7528\u6237 \u4e5f\u5c31\u662f\u4f18\u60e0\u5238id\u5728\u8fd9\u4e2a\u6d41\u7a0b\u4e2d\u662f\u786e\u5b9a\u7684 1.CouponTaskExecuteConsumer\u7c7b \u6267\u884conMessage\u6d88\u8d39mq\u4e2d\u7684\u6d88\u606f\uff0c\u83b7\u53d6\u5206\u53d1\u4efb\u52a1id \u7136\u540e\u6839\u636eid\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u8be2\u8be5\u4efb\u52a1 \u5224\u65ad\u4efb\u52a1\u72b6\u6001\u662f\u5426\u662f\u6267\u884c\u4e2d\uff0c\u5982\u679c\u4e0d\u662f\u6267\u884c\u4e2d\uff0c\u90a3\u5c31\u8bf4\u660e\u8fd8\u4e0d\u8be5\u6267\u884c \u5982\u679c\u4efb\u52a1\u662f\u6267\u884c\u4e2d\uff0c\u90a3\u5c31\u5224\u65ad\u4f18\u60e0\u5238\u72b6\u6001\u662f\u5426\u6b63\u786e\uff0c\u4e5f\u5c31\u662f\u770b\u4f18\u60e0\u5238\u662f\u751f\u6548\u8fd8\u662f\u7ed3\u675f \u5982\u679c\u4f18\u60e0\u5238\u751f\u6548\uff0c\u90a3\u5c31\u5f00\u59cb\u6267\u884c\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\/\u5206\u53d1\u4efb\u52a1 2.ReadExcelDistributionListener\u7c7b \u6bcf\u83b7\u53d6\u4e00\u884cExcel\u4e2d\u7684\u6570\u636e\u5c31\u4f1a\u6267\u884cinvoke\u4e00\u6b21 \u9996\u5148\u901a\u8fc7\u7f13\u5b58\u5224\u65ad\u4f18\u60e0\u5238\u6a21\u677f\u8bb0\u5f55\u5e93\u5b58\u662f\u5426\u5145\u8db3\uff0c\u56e0\u4e3a\u4e4b\u524d\u521b\u5efa\u4f18\u60e0\u5238\u7684\u65f6\u5019\u5df2\u7ecf\u5c06\u4fe1\u606f\u5b58\u5165redis \u5982\u679c\u7f13\u5b58\u4e2d\u7684\u5e93\u5b58-1\u4e4b\u540e\u5c0f\u4e8e0\uff0c\u90a3\u5c31\u8bf4\u660e\u6ca1\u5e93\u5b58\u4e86 \u8fc7\u4e86\u7f13\u5b58\u8fd9\u4e00\u5173\u5c31\u6263\u51cf\u6570\u636e\u5e93\u4e2d\u7684\u4f18\u60e0\u5238\u5e93\u5b58 \u4f18\u60e0\u5238\u5e93\u5b58\u6263\u51cf\u6210\u529f\u5c31\u8bf4\u660e\u53ef\u4ee5\u5206\u53d1\u7ed9\u7528\u6237\u4f18\u60e0\u5238\uff0c\u6240\u4ee5\u5f00\u59cb\u6784\u5efa\u7528\u6237\u9886\u5238\u8bb0\u5f55 \u8fd9\u91cc\u6ce8\u610f\uff0cvalidityPeriod\u662f\u7528\u6237\u83b7\u53d6\u5230\u4f18\u60e0\u5238\u4e4b\u540e\u4f18\u60e0\u5238\u7684\u4f7f\u7528\u6709\u6548\u671f\uff0c\u800c\u4e0d\u662f\u4f18\u60e0\u5238\u7684\u8fc7\u671f\u65f6\u95f4 \u6240\u4ee5\u8981\u901a\u8fc7\u5f53\u524d\u65f6\u95f4\u52a0\u4e0a\u8fd9\u4e2a\u504f\u79fb\u91cf\u83b7\u53d6\u8be5\u6761\u7528\u6237\u9886\u5238\u8bb0\u5f55\u7684\u8fc7\u671f\u65f6\u95f4 \u7136\u540e\u5c06\u7528\u6237\u9886\u5238\u8bb0\u5f55\u63d2\u5165\u6570\u636e\u5e93\u4e2d\uff0c\u901a\u8fc7\u552f\u4e00\u7d22\u5f15\u6821\u9a8c\u7528\u6237\u662f\u5426\u9886\u53d6\u8fc7\u8be5\u4f18\u60e0\u5238 \u7136\u540e\u6dfb\u52a0\u4f18\u60e0\u5238\u5230\u7528\u6237\u5df2\u9886\u53d6\u7684\u4f18\u60e0\u5238\u7f13\u5b58\u4e2d \u4f18\u60e0\u5238\u5206\u53d1\u5b8c\u6210\u4e4b\u540e\uff0c\u6267\u884cdoAfterAllAnalysed\u65b9\u6cd5\u8bbe\u7f6e\u8be5\u5206\u53d1\u4efb\u52a1\u7684\u72b6\u6001\u4e3a\u6267\u884c\u6210\u529f\u4ee5\u53ca\u5b8c\u6210\u65f6\u95f4<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u8bf7\u6559\u4e00\u4e0b\u9a6c\u54e5\uff0c\u8fd9\u91cc\u5bf9\u7f13\u5b58\u548c\u6570\u636e\u5e93\u7684\u64cd\u4f5c\u4e3a\u4ec0\u4e48\u4e0d\u52a0\u4e8b\u52a1\uff1f \u4f8b\u5982redis\u6263\u51cf\u5e93\u5b58\u6210\u529f\u4e86\uff0c\u4f46\u662f\u6570\u636e\u5e93\u64cd\u4f5c\u5931\u8d25\u4e86\u3002redis\u7684\u7f13\u5b58\u4e0d\u5f97\u56de\u6eda\u561b\uff1f<br>\u9a6c\u54e5\uff0c\u8fd8\u6709\u4e2a\u95ee\u9898\uff0c\u5373\u5728\u6d88\u8d39\u63a8\u9001\u4fe1\u606f\u7684\u65f6\u5019\uff0c\u4ee3\u7801\u4e2d\u662f\u5148\u68c0\u67e5\u6a21\u7248\u72b6\u6001\u5df2\u7ecf\u88ab\u53d6\u6d88\uff0c\u6ca1\u88ab\u53d6\u6d88\u624d\u5f80\u4e0b\u3002\u5982\u679c\u68c0\u67e5\u7684\u65f6\u5019\u6a21\u7248\u72b6\u6001\u662f\u6b63\u5728\u6267\u884c\u7684\uff0c\u68c0\u67e5\u903b\u8f91\u5b8c\u4e86\u4e4b\u540e\u6a21\u7248\u521a\u597d\u88ab\u53d6\u6d88\u4e86\u3002\u6b64\u65f6\u53c8\u5f80\u4e0b\u6267\u884c\u63a8\u9001\u4efb\u52a1\uff0c\u8fd9\u662f\u4e0d\u662f\u4e0d\u7b26\u5408\u903b\u8f91\uff1f<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u7b2c20\u5c0f\u8282\uff1a\u5f00\u53d1\u7528\u6237\u4f18\u60e0\u5238\u5206\u53d1\u529f\u80fd\uff08\u4e8c\uff09<\/h2>\n\n\n\n<h2 class=\"wp-block-heading\">\u4e1a\u52a1\u80cc\u666f<\/h2>\n\n\n\n<p>\u5728\u4e0a\u4e00\u8282\u4e2d\uff0c\u6211\u4eec\u901a\u8fc7 EasyExcel \u7684\u65b9\u5f0f\u8bfb\u53d6\u7528\u6237\u5206\u53d1\u5238 Excel\uff0c\u5e76\u5c06\u7ed3\u679c\u4fdd\u5b58\u5230\u4e86\u6570\u636e\u548c\u7f13\u5b58\u4e2d\u3002\u5982\u679c\u4f60\u521a\u5de5\u4f5c\uff0c\u5199\u4e4b\u524d\u7684\u4ee3\u7801\u6ca1\u95ee\u9898\uff0c\u5982\u679c\u5de5\u4f5c\u51e0\u5e74\u6216\u8005\u6709\u5168\u5c40\u610f\u8bc6\uff0c\u5927\u5bb6\u5c31\u5e94\u8be5\u80fd\u60f3\u5230\uff0c\u4ee3\u7801\u903b\u8f91\u662f\u6709\u95ee\u9898\u7684\u3002\u6211\u68b3\u7406\u4e86\u51e0\u70b9\u5185\u5bb9\uff0c\u5927\u5bb6\u4e00\u8d77\u770b\u770b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u6bcf\u6b21\u90fd\u5355\u6761\u65b9\u5f0f\u64cd\u4f5c Redis \u548c MySQL\uff0c\u7f51\u7edc\u6210\u672c\u6d88\u8017\u5de8\u5927\uff0c\u5e94\u8be5\u91c7\u7528\u6279\u91cf\u7684\u5f62\u5f0f\u6267\u884c\uff0c\u80fd\u8282\u7701\u5927\u91cf\u65f6\u95f4\u3002<\/li>\n\n\n\n<li>\u5982\u679c\u4e2d\u95f4\u5e94\u7528\u91cd\u542f\u6216\u8005\u5b95\u673a\uff0c\u6211\u4eec\u8fd8\u5f97\u91cd\u65b0\u6267\u884c\uff0c\u5e76\u4e0d\u7b26\u5408\u6211\u4eec\u7684\u9884\u671f\uff0c\u5e0c\u671b\u80fd\u591f\u4ece\u4e0a\u6b21\u6267\u884c\u70b9\u4f4d\u5f00\u59cb\u3002<\/li>\n\n\n\n<li>\u4e0d\u53ef\u80fd\u6240\u6709\u7528\u6237\u8bb0\u5f55\u90fd\u662f\u5206\u53d1\u4f18\u60e0\u5238\u6210\u529f\u7684\uff0c\u5982\u679c\u8bf4\u6709\u5206\u53d1\u5931\u8d25\u7684\u7528\u6237\u8bb0\u5f55\uff0c\u5e94\u8be5\u6807\u8bb0\u597d\u4fdd\u5b58\u6570\u636e\u5e93\u3002<\/li>\n\n\n\n<li>\u5c3d\u91cf\u4e0d\u8981\u5728\u5355\u4e2a\u6d88\u606f\u6d88\u8d39\u903b\u8f91\u4e2d\u6267\u884c\u592a\u957f\u7684\u4e1a\u52a1\u903b\u8f91\uff0c\u6211\u4eec\u5e94\u8be5\u5c3d\u53ef\u80fd\u7684\u8ba9\u8fd9\u4e2a\u6d88\u606f\u6267\u884c\u65f6\u957f\u53d8\u77ed\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u90a3\u8981\u4e48\u6709\u4e24\u79cd\u65b9\u6848\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li><strong>\u8981\u4e48\u5c31\u662f\u6211\u4eec\u628a\u8fd9\u4e2a\u6d88\u606f\u62c6\u6210 N \u4e2a\u5c0f\u7684\u6d88\u606f\u53bb\u8ba9\u5b83\u6267\u884c\u3002\u6bd4\u5982\u8bf4\u6211\u4e00\u767e\u4e07\u7684 Excel\uff0c\u6211\u53ef\u4ee5\u6bcf\u4e00\u6b21\u6d88\u606f\u8ba9\u5b83\u6267\u884c 5000\uff0c\u8fd9\u662f\u4e00\u4e2a\u903b\u8f91\u3002<\/strong>\u9009\u62e9\u4e86\u7b2c\u4e00\u4e2a<\/li>\n\n\n\n<li>\u7b2c\u4e8c\u5757\u7684\u8bdd\uff0c\u5c31\u662f\u628a\u5b83\u53d8\u6210\u4e00\u4e2a\u5f02\u6b65\u7684\u3001\u5b9a\u65f6\u89e6\u53d1\u7684\u903b\u8f91\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5176\u5b9e\u8fd9\u4e2a\u5f88\u597d\u5b9e\u73b0\uff0c\u5c31\u662f\u52a0\u4e00\u4e2a\u7f13\u5b58\uff0c\u91cc\u9762\u52a0\u4e00\u4e2a\u5b83\u5bf9\u5e94\u7684\u4e00\u4e2a\u8bb0\u5f55\u4f4d\u5c31\u53ef\u4ee5\u4e86\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u7b2c\u4e09\u6761\u7684\u8bdd\uff0c\u662f\u6211\u4eec\u8fd9\u91cc\u9762\u7684\u91cd\u5934\u620f\uff1a\u90a3\u5c31\u662f\u6211\u4eec\u6bcf\u6b21\u90fd\u4ee5\u5355\u6761\u7684\u5f62\u5f0f\u53bb\u64cd\u4f5c Redis \u548c\u6570\u636e\u5e93\uff0c\u7f51\u7edc\u6210\u672c\u662f\u5f88\u9ad8\u7684\u3002\u56e0\u4e3a\u50cf\u6211\u4eec\u5728\u6267\u884c\u8fd9\u4e2a\u903b\u8f91\u7684\u65f6\u5019\uff0c\u6700\u8017\u65f6\u7684\u5176\u5b9e\u662f IO\uff0c\u5bf9\u5427\uff1f\u5176\u5b9e CPU \u5b83\u662f\u4e0d\u74f6\u9888\u7684\u3002\u6211\u4eec\u4f1a\u91c7\u7528\u4e00\u79cd\u6279\u91cf\uff08Batch\uff09\u7684\u5f62\u5f0f\u53bb\u6267\u884c\uff0c\u80fd\u591f\u8282\u7701\u6211\u4eec\u5927\u91cf\u7684\u65f6\u95f4\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u7b2c\u56db\u5757\u7684\u8bdd\uff0c\u5728\u4e0a\u4e00\u4e2a\u7248\u672c\u91cc\u9762\uff0c\u6211\u4eec\u5982\u679c\u8bf4\u7528\u6237\u4ed6\u7684\u5206\u53d1\u5931\u8d25\u4e86\u7684\u8bdd\uff0c\u5176\u5b9e\u76f4\u63a5\u201c\u8df3\u8fc7\u201d\u4e86\uff0c\u5bf9\u5427\uff1f\u5728\u8fd9\u4e2a\u7248\u672c\u91cc\u9762\uff0c\u6211\u4eec\u662f\u8981\u7ed9\u5b83\u8bb0\u5f55\u4e0b\u6765\uff0c\u5e76\u4e14\u5728\u5e38\u89c4\u91cc\u9762\u7684\u8bdd\uff0c\u5728\u5e94\u7528\u754c\u9762\u91cc\u9762\u5b83\u662f\u80fd\u591f\u53bb\u8fdb\u884c\u5c55\u793a\u7684\uff0c\u6240\u4ee5\u8fd9\u4e2a\u662f\u5f88\u6709\u5fc5\u8981\u7684\u3002<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u7136\u540e\u4e0b\u9762\u662f\u6211\u4eec\u7b2c\u4e8c\u4e2a\u7248\u672c\u5b83\u7684\u4e00\u4e2a\u903b\u8f91\u56fe\u3002\u4e0a\u9762\u7684\u8bdd\u662f\u6211\u4eec\u4e4b\u524d\u8bb2\u8fc7\u7684\uff0c\u5c31\u662f\u4efb\u52a1\u7684\u6267\u884c\u9636\u6bb5\u3002\u5176\u5b9e\u91cd\u70b9\u662f\u6211\u4eec\u7684\u8bfb\u53d6\u5206\u53d1\u903b\u8f91\uff0c\u548c\u6211\u4eec\u5bf9\u5e94\u7684\u6267\u884c\u2014\u2014\u5c31\u662f\u53bb\u7ed9\u7528\u6237\u53bb\u8fdb\u884c\u6263\u51cf\u5e93\u5b58\uff0c\u4ee5\u53ca\u5bf9\u5e94\u7684\u4e00\u4e2a\u7ed9\u7528\u6237\u6dfb\u52a0\u4ed6\u5df2\u7ecf\u8981\u9886\u53d6\u7684\u4f18\u60e0\u5238\u3002\u8fd9\u4e00\u6b65\u903b\u8f91\uff0c\u8fd9\u4e24\u5757\u662f\u6bd4\u8f83\u91cd\u8981\u7684\u3002<br>v2 \u65b0\u7248\u672c\u4e1a\u52a1\u903b\u8f91\u5982\u4e0b\uff1a<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"1562\" width=\"1001\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/4aaf8180318471950d26ae870e2bc679.png\" alt=\"image-20240903181525512.png\"><\/p>\n\n\n\n<h2 class=\"wp-block-heading\" id=\"git\">Git \u5206\u652f<\/h2>\n\n\n\n<p>20240829_dev_coupon-distribute-v2_easyexcel-cache_ding.ma<br>origin\/20240831_dev_coupon-distribute-v2_easyexcel-cache_ding.ma(rebuild)<\/p>\n\n\n\n<p>v1\u90a3\u4e00\u5757\u8fd8\u53ea\u6709\u4e24\u70b9\u5206\u652f<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"651\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758355919-image-1024x651.png\" alt=\"\" class=\"wp-image-1302\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758355919-image-1024x651.png 1024w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758355919-image-300x191.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758355919-image-768x488.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758355919-image.png 1195w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/figure>\n\n\n\n<h2 class=\"wp-block-heading\">\u6570\u636e\u5e93\u8868\u8bbe\u8ba1<\/h2>\n\n\n\n<p>\u65b0\u589e\u7684\u8868\u7528\u4e8e\u8bb0\u5f55\u5206\u53d1\u8fc7\u7a0b\u4e2d\u56e0\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3\u6216\u7528\u6237\u5df2\u9886\u53d6\u7b49\u539f\u56e0\u5bfc\u81f4\u7684\u5931\u8d25\u8bb0\u5f55\u3002\u8fd9\u4e9b\u8bb0\u5f55\u4f1a\u751f\u6210\u9519\u8bef Excel \u6587\u4ef6\uff0c\u4ea4\u4ed8\u7ed9\u524d\u7aef\u5c55\u793a\uff0c\u65b9\u4fbf\u4e1a\u52a1\u4eba\u5458\u67e5\u770b\u5931\u8d25\u7528\u6237\u53ca\u5931\u8d25\u539f\u56e0\u3002<\/p>\n\n\n\n<p>\u8fdb\u5165 <code>one_coupon_rebuild_0<\/code> \u6570\u636e\u5e93\uff0c\u6267\u884c\u4e0b\u8ff0 SQL\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> CREATE TABLE `t_coupon_task_fail` (\n   `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT 'ID',\n   `batch_id` bigint(20) DEFAULT NULL COMMENT '\u6279\u6b21ID',\n   `json_object` text COMMENT '\u5931\u8d25\u5185\u5bb9',\n   PRIMARY KEY (`id`),\n   KEY `idx_batch_id` (`batch_id`) USING BTREE\n ) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COMMENT='\u4f18\u60e0\u5238\u6a21\u677f\u53d1\u9001\u4efb\u52a1\u5931\u8d25\u8bb0\u5f55\u8868';<\/code><\/pre>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5728 V2 \u91cc\u9762\u7684\u8bdd\uff0c\u6211\u4eec\u52a0\u4e86\u4e00\u5f20\u6570\u636e\u5e93\u8868\uff0c\u5b83\u5c31\u662f\u7528\u6765\u505a\u5206\u53d1\u5931\u8d25\u7684\u8bb0\u5f55\u4fdd\u5b58\u3002\u91cc\u9762\u5176\u5b9e\u5f88\u7b80\u5355\uff0c\u5c31\u662f\u4e00\u4e2a\u4e3b\u673a ID\u3001\u6211\u4eec\u7684\u4e00\u4e2a\u5206\u53d1\u4efb\u52a1 ID\uff0c\u8fd8\u6709\u4e00\u4e2a\u5931\u8d25\u7684\u5185\u5bb9\u3002\u5176\u5b9e\u5931\u8d25\u5185\u5bb9\u7684\u8bdd\u662f\u4e00\u4e2a JSON \u5b57\u7b26\u4e32\uff0c\u5b83\u6709\u4e24\u4e2a\u5b57\u6bb5\uff0c\u5f53\u7136\u540e\u7eed\u4e5f\u53ef\u4ee5\u6269\u5c55\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u5757\u7684\u8bdd\uff0c\u5c31\u662f\u5b83\u5f53\u524d\u7684\u4e00\u4e2a\u884c\u6570\uff0c\u4e5f\u5c31\u662f\u5728 Excel \u91cc\u9762\u7684\u884c\u6570\u3002<\/li>\n\n\n\n<li>\u7b2c\u4e8c\u5757\u7684\u8bdd\uff0c\u5c31\u662f\u5b83\u5931\u8d25\u539f\u56e0\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u6211\u4eec\u5931\u8d25\u539f\u56e0\u7684\u8bdd\u76ee\u524d\u5b9a\u4e49\u4e86\u4e24\u79cd\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u79cd\u7684\u8bdd\uff0c\u5c31\u662f\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3\uff1b<\/li>\n\n\n\n<li>\u7b2c\u4e8c\u5757\u7684\u8bdd\uff0c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u8fc7\u4f18\u60e0\u5238\u3002<\/li>\n<\/ul>\n<\/blockquote>\n\n\n\n<h2 class=\"wp-block-heading\">\u91cd\u6784\u4f18\u60e0\u5238\u5206\u53d1\u903b\u8f91<\/h2>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"493\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759329959-image-1024x493.png\" alt=\"\" class=\"wp-image-1374\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759329959-image-1024x493.png 1024w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759329959-image-300x144.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759329959-image-768x370.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759329959-image.png 1203w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/figure>\n\n\n\n<p>\u9996\u5148\u8001\u6837\u5b50\uff0c\u6211\u4eec\u53bb\u52a0\u4e00\u4e2a\u4f18\u60e0\u5238\u7684\u6a21\u677f\uff0c\u7136\u540e\u627e\u5230\u8fd9\u5f20\u8868\uff0c\u7136\u540e\u8bfb\u51fa\u6765\u3002\u7136\u540e\u6211\u4eec\u5728\u8fd9\u91cc\u53bb\u7ed9\u5b83\u521b\u5efa\u5bf9\u5e94\u7684\u4efb\u52a1\u3002\u6211\u4eec\u628a\u5bf9\u5e94\u7684\u53bb\u6253\u4e0a\u4e00\u4e9b\u65ad\u70b9\u2014\u2014\u56e0\u4e3a\u6211\u4eec\u8fd9\u4e2a\u5206\u652f\u6211\u8fd8\u6ca1\u6709\u63d0\u4ea4\uff0c\u8fd8\u5728\u8fdb\u884c\u4e00\u4e9b\u672c\u5730\u7684\u9a8c\u8bc1\uff0c\u770b\u770b\u662f\u5426\u4e07\u65e0\u4e00\u5931\uff0c\u907f\u514d\u540e\u7eed\u6709\u4e9b\u8fd4\u5de5\u3002\u7136\u540e\u56e0\u4e3a\u6211\u4eec\u7684\u5f88\u591a\u6838\u5fc3\u4ee3\u7801\u90fd\u5728\u5206\u53d1\u76d1\u542c\u5668\u91cc\u9762\uff0c\u6240\u4ee5\u8bf4\u6211\u4eec\u76f4\u63a5\u628a\u65ad\u70b9\u6253\u5230\u8fd9\u91cc\u5c31\u597d\u4e86\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"869\" height=\"588\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331035-image.png\" alt=\"\" class=\"wp-image-1375\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331035-image.png 869w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331035-image-300x203.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331035-image-768x520.png 768w\" sizes=\"auto, (max-width: 869px) 100vw, 869px\" \/><\/figure>\n\n\n\n<p>\u7b2c\u4e8c\u4e2a\u65ad\u70b9\uff1a\u7136\u540e\u6211\u4eec\u9700\u8981\u91cd\u542f\u4e00\u4e0b\u3002\u91cd\u542f\u7684\u8bdd\u5176\u5b9e\u4e0d\u5f71\u54cd\u6211\u4eec\u53d1\u9001\u3002\u903b\u8f91\u5df2\u7ecf\u6253\u5230\u8fd9\u8fb9\u4e86\uff0c\u7136\u540e\u5230\u6211\u4eec\u7684\u76d1\u542c\u5668\u8fd9\u91cc\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"829\" height=\"656\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331221-image.png\" alt=\"\" class=\"wp-image-1376\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331221-image.png 829w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331221-image-300x237.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759331221-image-768x608.png 768w\" sizes=\"auto, (max-width: 829px) 100vw, 829px\" \/><\/figure>\n\n\n\n<p>\u9996\u5148\u505c\u5728listener\u8fd9\u91cc\u3002\u521a\u624d\u6211\u4eec\u6709\u8bf4\uff0c\u5982\u679c\u8bf4\u5b95\u673a\u7684\u8bdd\uff0c\u5e0c\u671b\u6211\u4eec\u80fd\u591f\u5728\u5b83\u7684\u5b95\u673a\u524d\u7684\u70b9\u4f4d\u53bb\u7ee7\u7eed\u6267\u884c\u3002\u6240\u4ee5\u8bf4\u8fd9\u91cc\u6211\u4eec\u4f1a\u6709\u4e00\u4e2a\u5224\u65ad\uff1a\u6211\u4eec\u5728\u7f13\u5b58\u91cc\u9762\u52a0\u4e86\u4e00\u4e2a Key\uff0c\u7136\u540e\u8fd9\u4e2a Key \u8bb0\u4f4f\u4e86\u5c31\u662f\u6211\u4eec\u672c\u6b21\u4efb\u52a1\u6240\u5bf9\u5e94\u7684\u4e00\u4e2a\u6d88\u8d39\u70b9\u4f4d\u662f\u4ec0\u4e48\u3002\u5982\u679c\u8bf4\u6211\u5f53\u524d\u7684\u4e00\u4e2a\u8fdb\u5ea6\u662f\u5c0f\u4e8e\u8fd9\u4e2a\u503c\u7684\u8bdd\uff0c\u5c31\u8bc1\u660e\u5b83\u5b95\u673a\u8fc7\uff0c\u6211\u4eec\u76f4\u63a5return\u8df3\u8fc7\u5c31\u597d\u4e86\u3002\u7136\u540e\u5f00\u59cb\u6267\u884c\u3002\u4ee3\u7801\u5982\u4e0b<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Override\npublic void invoke(CouponTaskExcelObject data, AnalysisContext context) {\n    \n    Long couponTaskId = couponTaskD0.getid();\n     \n    \/\/ \u83b7\u53d6\u5f53\u524d\u8fdb\u5ea6\uff0c\u5224\u65ad\u662f\u5426\u5df2\u7ecf\u6267\u884c\u8fc7\u3002\u5982\u679c\u5df2\u6267\u884c\uff0c\u5219\u8df3\u8fc7\u5373\u53ef\uff0c\u9632\u6b62\u6267\u884c\u5230\u4e00\u534a\u5e94\u7528\u5b95\u673a\n    String templateTaskExecuteProgressKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_PROGRESS_KEY, couponTaskId);\n    String progress = stringRedisTemplate.opsForValue().get(templateTaskExecuteProgressKey);\n    if (StrUtil.isNotBlack(progress) &amp;&amp; Integer.parseInt(progress)\n        ++rowCount;\n        return;\n    }\n <\/code><\/pre>\n\n\n\n<p>\u7b2c\u4e8c\u6b65\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u8981\u53bb Redis \u91cc\u9762\u53bb\u8fdb\u884c\u6263\u51cf\u5bf9\u5e94\u7684\u4f18\u60e0\u5238\u6a21\u677f\u7684\u5e93\u5b58\u3002\u4e4b\u524d\u662f\u8fd9\u6837\u7684\uff0c\u73b0\u5728\u7684\u8bdd\uff0c\u56e0\u4e3a\u6211\u4eec\u4e0a\u9762\u6709\u8bb2\u8fc7\uff0c\u6211\u4eec\u5e0c\u671b\u6709\u4e00\u4e2a\u6279\u6b21\u7684\u6982\u5ff5\u2014\u2014\u6bd4\u5982\u8bf4\u6211\u4eec\u4e00\u4e2a\u6279\u6b21\u662f 5000\u3002\u5982\u679c\u8bf4\u5b83\u7684\u6279\u6b21\u4e0d\u8db3 5000 \u7684\u8bdd\uff0c\u6211\u4eec\u8981\u6709\u4e00\u4e2a\u7f13\u51b2\u961f\u5217\u7ed9\u5b83\u7f13\u5b58\u8d77\u6765\u3002<br>\u8fd9\u91cc\u7684\u8bdd\uff0c\u6211\u4eec\u91c7\u7528\u4e86\u5728 Redis \u91cc\u9762\u5f00\u8f9f\u4e86\u4e00\u4e2a\u96c6\u5408\uff0c\u8ba9\u5b83\u53bb\u8fdb\u884c\u4e00\u4e2a\u7f13\u5b58\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u83b7\u53d6 LUA \u811a\u672c\uff0c\u5e76\u4fdd\u5b58\u5230 Hutool \u7684\u5355\u4f8b\u7ba1\u7406\u5668\uff0c\u4e0b\u6b21\u76f4\u63a5\u83b7\u53d6\u4e0d\u9700\u8981\u52a0\u8f7d\nDefaultRedisScript&lt;Long> buildLuaScript = Singleton.get(STOCK_DECREMENT_AND_BATCH_SAVE_USER_RECORD_LUA_PATH, () -> {\n    DefaultRedisScript&lt;Long> redisScript = new DefaultRedisScript&lt;>();\n    redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(STOCK_DECREMENT_AND_BATCH_SAVE_USER_RECORD_LUA_PATH)));\n    redisScript.setResultType(Long.class);\n    return redisScript;\n});\n\n\/\/ \u6267\u884c LUA \u811a\u672c\u8fdb\u884c\u6263\u51cf\u5e93\u5b58\u4ee5\u53ca\u589e\u52a0 Redis \u7528\u6237\u9886\u5238\u8bb0\u5f55\nString couponTemplateKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, couponTemplateD0.getId());\nString batchUserSetKey = String.format(EngineRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, couponTaskId);\nMap&lt;Object, Object> userRowNumMap = MapUtil.builder()\n    .put(\"userId\", data.getUserId())\n    .put(\"rowNum\", v -> rowNumCount + 1)\n    .build();\nLong combinedFiled = stringRedisTemplate.execute(buildLuaScript, ListUtil.of(couponTemplateKey, batchUserSetKey), JSON.toJSONString(userRowNumMap));\n\n\/\/ firstField \u4e3a false \u8bf4\u660e\u4f18\u60e0\u5238\u5df2\u7ecf\u6ca1\u6709\u5e93\u5b58\u4e86\nboolean firstField = StockDecrementReturnUtil.extractFirstField(combinedFiled);<\/code><\/pre>\n\n\n\n<p>\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\uff1a\u8fd9\u91cc\u4e00\u822c\u6211\u5efa\u8bae\uff0c\u5982\u679c\u8bf4\u6d89\u53ca\u5230\u591a\u6b21\u5b83\u5bf9\u5e94\u7684 Redis \u64cd\u4f5c\uff0c\u6211\u4eec\u5c3d\u91cf\u80fd\u7528 Redis \u811a\u672c\u5c31\u53bb\u505a\u3002\u8fd9\u4e2a\u5efa\u8bae\u662f\uff1a\u5927\u5bb6\u518d\u53bb\u4e00\u4e9b\u6bd4\u8f83\u6838\u5fc3\u3001\u6216\u8005\u8bf4\u4e00\u4e9b ToC\uff08\u9762\u5411\u7528\u6237\uff09\u3001\u53d8\u5316\u9891\u7387\u6bd4\u8f83\u9ad8\u7684\u7cfb\u7edf\u91cc\u9762\uff0c\u6211\u5efa\u8bae\u5927\u5bb6\u8fd9\u4e48\u505a\u3002\u56e0\u4e3a\u8fd9\u6837\u7684\u8bdd\u4f1a\u8282\u7701\u4e00\u6b21\u7f51\u7edc\u7684 IO\u3002<\/p>\n\n\n\n<p>\u5bf9\uff0c\u53ef\u80fd\u4e0d\u6b62\u4e00\u6b21\u2014\u2014\u770b\u4f60 Redis \u811a\u672c\u91cc\u9762\u4f1a\u6267\u884c\u591a\u5c11\u903b\u8f91\u3002<br>\u7136\u540e\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\u8fd9\u4e00\u5757\u662f\u6211\u4eec build \u7684\u4e00\u4e2a Redis \u811a\u672c\u3002\u6211\u4eec\u7684 Keys \u7b2c\u4e00\u4e2a\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u4f18\u60e0\u5238\u6a21\u677f\u7684 Key\uff1b\u7b2c\u4e8c\u4e2a\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u6784\u5efa\u7684\u7f13\u51b2\u961f\u5217\u7684\u4e00\u4e2a Key\u3002\u7136\u540e\u6211\u4eec\u5bf9\u5e94\u7684\u53c2\u6570\u8fd9\u4e00\u5757\uff0c\u6211\u4eec\u53bb\u628a\u5b83\u5f53\u524d\u7684\u7528\u6237 ID \u4ee5\u53ca\u5b83\u6240\u5728\u7684\u884c\u6570\u4f20\u8fdb\u53bb\u3002<\/p>\n\n\n\n<p>\u5927\u5bb6\u53ef\u80fd\u4f1a\u7ea0\u7ed3\uff1a\u6211\u7b2c\u4e00\u4e2a\u6d88\u606f\u4e3a\u4ec0\u4e48 <code>rowNumber<\/code> \u4f1a\u662f 2\uff1f\u56e0\u4e3a\u6211\u4eec\u5728 Excel \u91cc\u9762\u7684\u7b2c\u4e00\u884c\u662f\u6807\u9898\u884c\uff0c\u4f46\u662f\u6211\u4eec\u7684\u7528\u6237\u7684\u7b2c\u4e00\u884c\u8bb0\u5f55\u5176\u5b9e\u5728 Excel \u91cc\u9762\u5b83\u7684\u884c\u6570\u662f 2\u3002\u6240\u4ee5\u8bf4\u6211\u4eec\u7ed9\u5b83\u5728\u6267\u884c\u7684\u65f6\u5019\u5148\u52a0\u4e86\u4e00\u4e2a 1\u3002<\/p>\n\n\n\n<p>\u56e0\u4e3a\u4ee3\u7801\u903b\u8f91\u6574\u4f53\u6bd4\u8f83\u8026\u5408\uff0c\u6240\u4ee5\u6211\u4eec\u5c06\u6574\u4f53\u4ee3\u7801\u5949\u4e0a\uff0c\u5e76\u5728\u63a5\u4e0b\u6765\u7684\u5c0f\u8282\u4e2d\u9010\u4e2a\u89e3\u6790\u5176\u4e2d\u7684\u6838\u5fc3\u4e1a\u52a1\u903b\u8f91\u548c\u4f18\u5316\u3002\u5176\u4e2d\u4f18\u5316\u903b\u8f91\u4e3b\u8981\u6d89\u53ca\u4e24\u4e2a\u5b8c\u6574\u7c7b\uff1a<code>ReadExcelDistributionListener<\/code> \u548c <code>CouponExecuteDistributionConsumer<\/code>\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5176\u4e2d\u4e00\u4e9b\u7ec6\u8282\u5904\u4f7f\u7528\u4e86\u8f83\u591a\u5c0f\u6280\u5de7\uff0c\u770b\u4e0b\u6587\u7ec6\u7ec6\u9053\u6765\u3002<\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1-read-excel-distribution-listener\">1. ReadExcelDistributionListener<\/h3>\n\n\n\n<pre class=\"wp-block-code\"><code>@RequiredArgsConstructor\n public class ReadExcelDistributionListener extends AnalysisEventListener&lt;CouponTaskExcelObject&gt; {\n \u200b\n     private final CouponTaskDO couponTaskDO;\n     private final CouponTemplateDO couponTemplateDO;\n     private final CouponTaskFailMapper couponTaskFailMapper;\n \u200b\n     private final StringRedisTemplate stringRedisTemplate;\n     private final CouponExecuteDistributionProducer couponExecuteDistributionProducer;\n \u200b\n     private int rowCount = 1;\n     private final static String STOCK_DECREMENT_AND_BATCH_SAVE_USER_RECORD_LUA_PATH = \"lua\/stock_decrement_and_batch_save_user_record.lua\";\n     private final static int BATCH_USER_COUPON_SIZE = 5000;\n \u200b\n     @Override\n     public void invoke(CouponTaskExcelObject data, AnalysisContext context) {\n         Long couponTaskId = couponTaskDO.getId();\n \u200b\n         \/\/ \u83b7\u53d6\u5f53\u524d\u8fdb\u5ea6\uff0c\u5224\u65ad\u662f\u5426\u5df2\u7ecf\u6267\u884c\u8fc7\u3002\u5982\u679c\u5df2\u6267\u884c\uff0c\u5219\u8df3\u8fc7\u5373\u53ef\uff0c\u9632\u6b62\u6267\u884c\u5230\u4e00\u534a\u5e94\u7528\u5b95\u673a\n         String templateTaskExecuteProgressKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_PROGRESS_KEY, couponTaskId);\n         String progress = stringRedisTemplate.opsForValue().get(templateTaskExecuteProgressKey);\n         if (StrUtil.isNotBlank(progress) &amp;&amp; Integer.parseInt(progress) &gt;= rowCount) {\n             ++rowCount;\n             return;\n         }\n \u200b\n         \/\/ \u83b7\u53d6 LUA \u811a\u672c\uff0c\u5e76\u4fdd\u5b58\u5230 Hutool \u7684\u5355\u4f8b\u7ba1\u7406\u5bb9\u5668\uff0c\u4e0b\u6b21\u76f4\u63a5\u83b7\u53d6\u4e0d\u9700\u8981\u52a0\u8f7d\n         DefaultRedisScript&lt;Long&gt; buildLuaScript = Singleton.get(STOCK_DECREMENT_AND_BATCH_SAVE_USER_RECORD_LUA_PATH, () -&gt; {\n             DefaultRedisScript&lt;Long&gt; redisScript = new DefaultRedisScript&lt;&gt;();\n             redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(STOCK_DECREMENT_AND_BATCH_SAVE_USER_RECORD_LUA_PATH)));\n             redisScript.setResultType(Long.class);\n             return redisScript;\n         });\n \u200b\n         \/\/ \u6267\u884c LUA \u811a\u672c\u8fdb\u884c\u6263\u51cf\u5e93\u5b58\u4ee5\u53ca\u589e\u52a0 Redis \u7528\u6237\u9886\u5238\u8bb0\u5f55\n         String couponTemplateKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, couponTemplateDO.getId());\n         String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, couponTaskId);\n         Map&lt;Object, Object&gt; userRowNumMap = MapUtil.builder()\n                 .put(\"userId\", data.getUserId())\n                 .put(\"rowNum\", rowCount + 1)\n                 .build();\n         Long combinedFiled = stringRedisTemplate.execute(buildLuaScript, ListUtil.of(couponTemplateKey, batchUserSetKey), JSON.toJSONString(userRowNumMap));\n \u200b\n         \/\/ firstField \u4e3a false \u8bf4\u660e\u4f18\u60e0\u5238\u5df2\u7ecf\u6ca1\u6709\u5e93\u5b58\u4e86\n         boolean firstField = StockDecrementReturnCombinedUtil.extractFirstField(combinedFiled);\n         if (!firstField) {\n             \/\/ \u540c\u6b65\u5f53\u524d\u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n             stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n             ++rowCount;\n \u200b\n             \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n             Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n                     .put(\"rowNum\", rowCount + 1)\n                     .put(\"cause\", \"\u4f18\u60e0\u5238\u6a21\u677f\u65e0\u5e93\u5b58\")\n                     .build();\n             CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n                     .batchId(couponTaskDO.getBatchId())\n                     .jsonObject(JSON.toJSONString(objectMap, SerializerFeature.WriteMapNullValue))\n                     .build();\n             couponTaskFailMapper.insert(couponTaskFailDO);\n             return;\n         }\n \u200b\n         \/\/ \u83b7\u53d6\u7528\u6237\u9886\u5238\u96c6\u5408\u957f\u5ea6\n         int batchUserSetSize = StockDecrementReturnCombinedUtil.extractSecondField(combinedFiled.intValue());\n \u200b\n         \/\/ batchUserSetSize = BATCH_USER_COUPON_SIZE \u65f6\u53d1\u9001\u6d88\u606f\u6d88\u8d39\uff0c\u4e0d\u6ee1\u8db3\u6761\u4ef6\u4ec5\u8bb0\u5f55\u6267\u884c\u8fdb\u5ea6\u5373\u53ef\n         if (batchUserSetSize &lt; BATCH_USER_COUPON_SIZE) {\n             \/\/ \u540c\u6b65\u5f53\u524d Excel \u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n             stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n             ++rowCount;\n             return;\n         }\n \u200b\n         CouponTemplateDistributionEvent couponTemplateDistributionEvent = CouponTemplateDistributionEvent.builder()\n                 .couponTaskId(couponTaskId)\n                 .shopNumber(couponTaskDO.getShopNumber())\n                 .couponTemplateId(couponTemplateDO.getId())\n                 .couponTaskBatchId(couponTaskDO.getBatchId())\n                 .couponTemplateConsumeRule(couponTemplateDO.getConsumeRule())\n                 .batchUserSetSize(batchUserSetSize)\n                 .distributionEndFlag(Boolean.FALSE)\n                 .build();\n         couponExecuteDistributionProducer.sendMessage(couponTemplateDistributionEvent);\n \u200b\n         \/\/ \u540c\u6b65\u5f53\u524d\u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n         stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n         ++rowCount;\n     }\n \u200b\n     @Override\n     public void doAfterAllAnalysed(AnalysisContext context) {\n         \/\/ \u53d1\u9001 Excel \u89e3\u6790\u5b8c\u6210\u6807\u8bc6\uff0c\u5373\u4f7f\u4e0d\u6ee1\u8db3\u6279\u91cf\u4fdd\u5b58\u7684\u6570\u91cf\u4e5f\u5f97\u4fdd\u5b58\u5230\u6570\u636e\u5e93\n         CouponTemplateDistributionEvent couponTemplateExecuteEvent = CouponTemplateDistributionEvent.builder()\n                 .distributionEndFlag(Boolean.TRUE) \/\/ \u8bbe\u7f6e\u89e3\u6790\u5b8c\u6210\u6807\u8bc6\n                 .shopNumber(couponTaskDO.getShopNumber())\n                 .couponTemplateId(couponTemplateDO.getId())\n                 .couponTemplateConsumeRule(couponTemplateDO.getConsumeRule())\n                 .couponTaskBatchId(couponTaskDO.getBatchId())\n                 .couponTaskId(couponTaskDO.getId())\n                 .build();\n         couponExecuteDistributionProducer.sendMessage(couponTemplateExecuteEvent);\n     }\n }<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2-coupon-execute-distribution-consumer\">2. CouponExecuteDistributionConsumer<\/h3>\n\n\n\n<pre class=\"wp-block-code\"><code> @Component\n @RequiredArgsConstructor\n @RocketMQMessageListener(\n         topic = \"one-coupon_distribution-service_coupon-execute-distribution_topic${unique-name:}\",\n         consumerGroup = \"one-coupon_distribution-service_coupon-execute-distribution_cg${unique-name:}\"\n )\n @Slf4j(topic = \"CouponExecuteDistributionConsumer\")\n public class CouponExecuteDistributionConsumer implements RocketMQListener&lt;MessageWrapper&lt;CouponTemplateDistributionEvent&gt;&gt; {\n \u200b\n     private final UserCouponMapper userCouponMapper;\n     private final CouponTemplateMapper couponTemplateMapper;\n     private final CouponTaskMapper couponTaskMapper;\n     private final CouponTaskFailMapper couponTaskFailMapper;\n     private final StringRedisTemplate stringRedisTemplate;\n \u200b\n     @Lazy\n     @Autowired\n     private CouponExecuteDistributionConsumer couponExecuteDistributionConsumer;\n \u200b\n     private final static int BATCH_USER_COUPON_SIZE = 5000;\n     private static final String BATCH_SAVE_USER_COUPON_LUA_PATH = \"lua\/batch_user_coupon_list.lua\";\n \u200b\n     @Transactional(rollbackFor = Exception.class)\n     @Override\n     public void onMessage(MessageWrapper&lt;CouponTemplateDistributionEvent&gt; messageWrapper) {\n         \/\/ \u5f00\u5934\u6253\u5370\u65e5\u5fd7\uff0c\u5e73\u5e38\u53ef Debug \u770b\u4efb\u52a1\u53c2\u6570\uff0c\u7ebf\u4e0a\u53ef\u62a5\u5e73\u5b89\uff08\u6bd4\u5982\u6d88\u606f\u662f\u5426\u6d88\u8d39\uff0c\u91cd\u65b0\u6295\u9012\u65f6\u83b7\u53d6\u53c2\u6570\u7b49\uff09\n         log.info(\"&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u4efb\u52a1\u6267\u884c\u63a8\u9001@\u5206\u53d1\u5230\u7528\u6237\u8d26\u53f7 - \u6267\u884c\u6d88\u8d39\u903b\u8f91\uff0c\u6d88\u606f\u4f53\uff1a{}\", JSON.toJSONString(messageWrapper));\n \u200b\n         \/\/ \u5f53\u4fdd\u5b58\u7528\u6237\u4f18\u60e0\u5238\u96c6\u5408\u8fbe\u5230\u6279\u91cf\u4fdd\u5b58\u6570\u91cf\n         CouponTemplateDistributionEvent event = messageWrapper.getMessage();\n         if (!event.getDistributionEndFlag() &amp;&amp; event.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0) {\n             decrementCouponTemplateStockAndSaveUserCouponList(event);\n             return;\n         }\n \u200b\n         \/\/ \u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6\u4e3a TRUE\uff0c\u4ee3\u8868\u5df2\u7ecf\u6ca1\u6709 Excel \u8bb0\u5f55\u4e86\n         if (event.getDistributionEndFlag()) {\n             String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n             Long batchUserIdsSize = stringRedisTemplate.opsForSet().size(batchUserSetKey);\n             event.setBatchUserSetSize(batchUserIdsSize.intValue());\n \u200b\n             decrementCouponTemplateStockAndSaveUserCouponList(event);\n             List&lt;String&gt; batchUserMaps = stringRedisTemplate.opsForSet().pop(batchUserSetKey, Integer.MAX_VALUE);\n             \/\/ \u6b64\u65f6\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\u5982\u679c\u8fd8\u6709\u503c\uff0c\u5c31\u610f\u5473\u7740\u53ef\u80fd\u5e93\u5b58\u4e0d\u8db3\u5f15\u8d77\u7684\n             if (CollUtil.isNotEmpty(batchUserMaps)) {\n                 \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n                 List&lt;CouponTaskFailDO&gt; couponTaskFailDOList = new ArrayList&lt;&gt;(batchUserMaps.size());\n                 for (String batchUserMapStr : batchUserMaps) {\n                     Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n                             .put(\"rowNum\", JSON.parseObject(batchUserMapStr).get(\"rowNum\"))\n                             .put(\"cause\", \"\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u4e0d\u8db3\")\n                             .build();\n                     CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n                             .batchId(event.getCouponTaskBatchId())\n                             .jsonObject(com.alibaba.fastjson.JSON.toJSONString(objectMap))\n                             .build();\n                     couponTaskFailDOList.add(couponTaskFailDO);\n                 }\n \u200b\n                 \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\n                 couponTaskFailMapper.insert(couponTaskFailDOList);\n             }\n \u200b\n             \/\/ \u786e\u4fdd\u6240\u6709\u7528\u6237\u90fd\u5df2\u7ecf\u63a5\u5230\u4f18\u60e0\u5238\u540e\uff0c\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u5b8c\u6210\u65f6\u95f4\n             CouponTaskDO couponTaskDO = CouponTaskDO.builder()\n                     .id(event.getCouponTaskId())\n                     .status(CouponTaskStatusEnum.SUCCESS.getStatus())\n                     .completionTime(new Date())\n                     .build();\n             couponTaskMapper.updateById(couponTaskDO);\n         }\n     }\n \u200b\n     @SneakyThrows\n     private void decrementCouponTemplateStockAndSaveUserCouponList(CouponTemplateDistributionEvent event) {\n         \/\/ \u5982\u679c\u7b49\u4e8e 0 \u610f\u5473\u7740\u5df2\u7ecf\u6ca1\u6709\u4e86\u5e93\u5b58\uff0c\u76f4\u63a5\u8fd4\u56de\u5373\u53ef\n         Integer couponTemplateStock = decrementCouponTemplateStock(event, event.getBatchUserSetSize());\n         if (couponTemplateStock &lt;= 0) {\n             return;\n         }\n \u200b\n         \/\/ \u83b7\u53d6 Redis \u4e2d\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\n         String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n         List&lt;String&gt; batchUserMaps = stringRedisTemplate.opsForSet().pop(batchUserSetKey, couponTemplateStock);\n \u200b\n         \/\/ \u56e0\u4e3a batchUserIds \u6570\u636e\u8f83\u591a\uff0cArrayList \u4f1a\u8fdb\u884c\u6570\u6b21\u6269\u5bb9\uff0c\u4e3a\u4e86\u907f\u514d\u989d\u5916\u6027\u80fd\u6d88\u8017\uff0c\u76f4\u63a5\u521d\u59cb\u5316 batchUserIds \u5927\u5c0f\u7684\u6570\u7ec4\n         List&lt;UserCouponDO&gt; userCouponDOList = new ArrayList&lt;&gt;(batchUserMaps.size());\n         Date now = new Date();\n \u200b\n         \/\/ \u6784\u5efa userCouponDOList \u7528\u6237\u4f18\u60e0\u5238\u6279\u91cf\u6570\u7ec4\n         for (String each : batchUserMaps) {\n             JSONObject userIdAndRowNumJsonObject = JSON.parseObject(each);\n             DateTime validEndTime = DateUtil.offsetHour(now, JSON.parseObject(event.getCouponTemplateConsumeRule()).getInteger(\"validityPeriod\"));\n             UserCouponDO userCouponDO = UserCouponDO.builder()\n                     .id(IdUtil.getSnowflakeNextId())\n                     .couponTemplateId(event.getCouponTemplateId())\n                     .rowNum(userIdAndRowNumJsonObject.getInteger(\"rowNum\"))\n                     .userId(userIdAndRowNumJsonObject.getLong(\"userId\"))\n                     .receiveTime(now)\n                     .receiveCount(1) \/\/ \u4ee3\u8868\u7b2c\u4e00\u6b21\u9886\u53d6\u8be5\u4f18\u60e0\u5238\n                     .validStartTime(now)\n                     .validEndTime(validEndTime)\n                     .source(CouponSourceEnum.PLATFORM.getType())\n                     .status(CouponStatusEnum.EFFECTIVE.getType())\n                     .createTime(new Date())\n                     .updateTime(new Date())\n                     .delFlag(0)\n                     .build();\n             userCouponDOList.add(userCouponDO);\n         }\n \u200b\n         \/\/ \u5e73\u53f0\u4f18\u60e0\u5238\u6bcf\u4e2a\u7528\u6237\u9650\u9886\u4e00\u6b21\u3002\u6279\u91cf\u65b0\u589e\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\uff0c\u5e95\u5c42\u901a\u8fc7\u9012\u5f52\u65b9\u5f0f\u76f4\u5230\u5168\u90e8\u65b0\u589e\u6210\u529f\n         batchSaveUserCouponList(event.getCouponTemplateId(), event.getCouponTaskBatchId(), userCouponDOList);\n \u200b\n         \/\/ \u5c06\u8fd9\u4e9b\u4f18\u60e0\u5238\u6dfb\u52a0\u5230\u7528\u6237\u7684\u9886\u5238\u8bb0\u5f55\u4e2d\n         List&lt;String&gt; userIdList = userCouponDOList.stream()\n                 .map(UserCouponDO::getUserId)\n                 .map(String::valueOf)\n                 .toList();\n         String userIdsJson = new ObjectMapper().writeValueAsString(userIdList);\n \u200b\n         List&lt;String&gt; couponIdList = userCouponDOList.stream()\n                 .map(each -&gt; StrUtil.builder()\n                         .append(event.getCouponTemplateId())\n                         .append(\"_\")\n                         .append(each.getId())\n                         .toString())\n                 .map(String::valueOf)\n                 .toList();\n         String couponIdsJson = new ObjectMapper().writeValueAsString(couponIdList);\n \u200b\n         \/\/ \u8c03\u7528 Lua \u811a\u672c\u65f6\uff0c\u4f20\u9012\u53c2\u6570\n         List&lt;String&gt; keys = Arrays.asList(\n                 \/\/ \u4e3a\u4ec0\u4e48\u8981\u8fdb\u884c\u66ff\u6362 %s \u4e3a\u7a7a\u767d\u5b57\u7b26\u4e32\uff1f\u56e0\u4e3a\u540e\u7eed\u4ee3\u7801\u9700\u8981\u4f7f\u7528 %s \u8fdb\u884c\u52a8\u6001\u503c\u66ff\u6362\uff0c\u4f46\u662f\u5f53\u524d LUA \u811a\u672c\u4e2d\u4e0d\u9700\u8981\uff0c\u6240\u4ee5\u4e3a\u4e86\u517c\u5bb9\u540e\u7eed\u4e0d\u6539\u52a8\u7279\u6b64\u66ff\u6362\n                 StrUtil.replace(EngineRedisConstant.USER_COUPON_TEMPLATE_LIST_KEY, \"%s\", \"\"),\n                 USER_COUPON_TEMPLATE_LIMIT_KEY,\n                 String.valueOf(event.getCouponTemplateId())\n         );\n         List&lt;String&gt; args = Arrays.asList(\n                 userIdsJson,\n                 couponIdsJson,\n                 String.valueOf(new Date().getTime()),\n                 String.valueOf(event.getValidEndTime().getTime())\n         );\n \u200b\n         \/\/ \u83b7\u53d6 LUA \u811a\u672c\uff0c\u5e76\u4fdd\u5b58\u5230 Hutool \u7684\u5355\u4f8b\u7ba1\u7406\u5bb9\u5668\uff0c\u4e0b\u6b21\u76f4\u63a5\u83b7\u53d6\u4e0d\u9700\u8981\u52a0\u8f7d\n         DefaultRedisScript&lt;Void&gt; buildLuaScript = Singleton.get(BATCH_SAVE_USER_COUPON_LUA_PATH, () -&gt; {\n             DefaultRedisScript&lt;Void&gt; redisScript = new DefaultRedisScript&lt;&gt;();\n             redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(BATCH_SAVE_USER_COUPON_LUA_PATH)));\n             redisScript.setResultType(Void.class);\n             return redisScript;\n         });\n         stringRedisTemplate.execute(buildLuaScript, keys, args.toArray());\n       \n         \/\/ \u589e\u52a0\u5e93\u5b58\u56de\u6eda\u65b9\u6848\uff0c\u5982\u679c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u4f18\u60e0\u5238\u88ab\u6821\u9a8c\uff0c\u9700\u8981\u5c06 Redis \u9884\u6263\u51cf\u5e93\u5b58\u56de\u6eda\n         int originalUserCouponSize = batchUserMaps.size();\n         \/\/ \u5982\u679c\u7528\u6237\u5df2\u9886\u53d6\u88ab\u6821\u9a8c\u4f1a\u4ece\u96c6\u5408\u4e2d\u5220\u9664\n         int availableUserCouponSize = userCouponDOList.size();\n         int rollbackStock = originalUserCouponSize - availableUserCouponSize;\n         if (rollbackStock &gt; 0) {\n             \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u5e93\u5b58\u6570\u91cf\n             stringRedisTemplate.opsForHash().increment(\n                     String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, event.getCouponTemplateId()),\n                     \"stock\",\n                     rollbackStock\n             );\n \u200b\n             \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u5e93\u5e93\u5b58\u6570\u91cf\n             couponTemplateMapper.incrementCouponTemplateStock(event.getShopNumber(), event.getCouponTemplateId(), rollbackStock);\n         }\n     }\n \u200b\n     private Integer decrementCouponTemplateStock(CouponTemplateDistributionEvent event, Integer decrementStockSize) {\n         Long couponTemplateId = event.getCouponTemplateId();\n         int decremented = couponTemplateMapper.decrementCouponTemplateStock(event.getShopNumber(), couponTemplateId, decrementStockSize);\n \u200b\n         \/\/ \u5982\u679c\u4fee\u6539\u8bb0\u5f55\u5931\u8d25\uff0c\u610f\u5473\u7740\u4f18\u60e0\u5238\u5e93\u5b58\u5df2\u4e0d\u8db3\uff0c\u9700\u8981\u91cd\u8bd5\u83b7\u53d6\u5230\u53ef\u81ea\u51cf\u7684\u5e93\u5b58\u6570\u503c\n         if (!SqlHelper.retBool(decremented)) {\n             LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                     .eq(CouponTemplateDO::getShopNumber, event.getShopNumber())\n                     .eq(CouponTemplateDO::getId, couponTemplateId);\n             CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n             return decrementCouponTemplateStock(event, couponTemplateDO.getStock());\n         }\n \u200b\n         return decrementStockSize;\n     }\n \u200b\n     private void batchSaveUserCouponList(Long couponTemplateId, Long couponTaskBatchId, List&lt;UserCouponDO&gt; userCouponDOList) {\n         \/\/ MyBatis-Plus \u6279\u91cf\u6267\u884c\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\n         try {\n             userCouponMapper.insert(userCouponDOList, userCouponDOList.size());\n         } catch (Exception ex) {\n             Throwable cause = ex.getCause();\n             if (cause instanceof BatchExecutorException) {\n                 \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n                 List&lt;CouponTaskFailDO&gt; couponTaskFailDOList = new ArrayList&lt;&gt;();\n                 List&lt;UserCouponDO&gt; toRemove = new ArrayList&lt;&gt;();\n \u200b\n                 \/\/ \u8c03\u7528\u6279\u91cf\u65b0\u589e\u5931\u8d25\u540e\uff0c\u4e3a\u4e86\u907f\u514d\u5927\u91cf\u91cd\u590d\u5931\u8d25\uff0c\u6211\u4eec\u901a\u8fc7\u65b0\u589e\u5355\u6761\u8bb0\u5f55\u65b9\u5f0f\u6267\u884c\n                 userCouponDOList.forEach(each -&gt; {\n                     try {\n                         userCouponMapper.insert(each);\n                     } catch (Exception ignored) {\n                         Boolean hasReceived = couponExecuteDistributionConsumer.hasUserReceivedCoupon(couponTemplateId, each.getUserId());\n                         if (hasReceived) {\n                             \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n                             Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n                                     .put(\"rowNum\", each.getRowNum())\n                                     .put(\"cause\", \"\u7528\u6237\u5df2\u9886\u53d6\u8be5\u4f18\u60e0\u5238\")\n                                     .build();\n                             CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n                                     .batchId(couponTaskBatchId)\n                                     .jsonObject(com.alibaba.fastjson.JSON.toJSONString(objectMap))\n                                     .build();\n                             couponTaskFailDOList.add(couponTaskFailDO);\n \u200b\n                             \/\/ \u4ece userCouponDOList \u4e2d\u5220\u9664\u5df2\u7ecf\u5b58\u5728\u7684\u8bb0\u5f55\n                             toRemove.add(each);\n                         }\n                     }\n                 });\n \u200b\n                 \/\/ \u6279\u91cf\u65b0\u589e t_coupon_task_fail \u8868\n                 couponTaskFailMapper.insert(couponTaskFailDOList, couponTaskFailDOList.size());\n \u200b\n                 \/\/ \u5220\u9664\u5df2\u7ecf\u91cd\u590d\u7684\u5185\u5bb9\n                 userCouponDOList.removeAll(toRemove);\n                 return;\n             }\n \u200b\n             throw ex;\n         }\n     }\n \u200b\n     \/**\n      * \u67e5\u8be2\u7528\u6237\u662f\u5426\u5df2\u7ecf\u9886\u53d6\u8fc7\u4f18\u60e0\u5238\n      *\n      * @param couponTemplateId \u4f18\u60e0\u5238\u6a21\u677f ID\n      * @param userId           \u7528\u6237 ID\n      * @return \u7528\u6237\u4f18\u60e0\u5238\u6a21\u677f\u9886\u53d6\u4fe1\u606f\u662f\u5426\u5df2\u5b58\u5728\n      *\/\n     @Transactional(propagation = Propagation.NOT_SUPPORTED)\n     public Boolean hasUserReceivedCoupon(Long couponTemplateId, Long userId) {\n         LambdaQueryWrapper&lt;UserCouponDO&gt; queryWrapper = Wrappers.lambdaQuery(UserCouponDO.class)\n                 .eq(UserCouponDO::getUserId, userId)\n                 .eq(UserCouponDO::getCouponTemplateId, couponTemplateId);\n         return userCouponMapper.selectOne(queryWrapper) != null;\n     }\n }\n<\/code><\/pre>\n\n\n\n<h2 class=\"wp-block-heading\">\u8bb0\u5f55\u5206\u53d1\u70b9\u4f4d<\/h2>\n\n\n\n<p>\u5728\u6d88\u8d39\u8005\u8bfb\u53d6 Excel \u6267\u884c\u5230\u6700\u540e\u4e00\u6761\u8bb0\u5f55\u65f6\uff0c\u5982\u679c\u53d1\u751f\u5b95\u673a\uff0c\u6d88\u606f\u961f\u5217\u5728\u91cd\u8bd5\u65f6\u9700\u8981\u91cd\u65b0\u6267\u884c\u6574\u4e2a\u4efb\u52a1\uff0c\u8fd9\u663e\u7136\u4f1a\u9020\u6210\u65f6\u95f4\u548c\u8d44\u6e90\u7684\u6d6a\u8d39\u3002\u4e3a\u4e86\u4f18\u5316\u8fd9\u4e00\u70b9\uff0c\u6211\u4eec\u5728\u6bcf\u6b21\u6210\u529f\u5206\u53d1\u8bb0\u5f55\u540e\uff0c\u4f1a\u5c06\u8fdb\u5ea6\u4fdd\u5b58\u5230\u7f13\u5b58\u4e2d\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u83b7\u53d6\u5f53\u524d\u8fdb\u5ea6\uff0c\u5224\u65ad\u662f\u5426\u5df2\u7ecf\u6267\u884c\u8fc7\u3002\u5982\u679c\u5df2\u6267\u884c\uff0c\u5219\u8df3\u8fc7\u5373\u53ef\uff0c\u9632\u6b62\u6267\u884c\u5230\u4e00\u534a\u5e94\u7528\u5b95\u673a\n String templateTaskExecuteProgressKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_PROGRESS_KEY, couponTaskId);\n String progress = stringRedisTemplate.opsForValue().get(templateTaskExecuteProgressKey);\n if (StrUtil.isNotBlank(progress) &amp;&amp; Integer.parseInt(progress) &gt;= rowCount) {\n     ++rowCount;\n     return;\n }\n \u200b\n \/\/ .....\n \u200b\n \/\/ \u8bb0\u5f55\u6267\u884c\u540e\uff0c\u540c\u6b65\u5f53\u524d\u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n ++rowCount;<\/code><\/pre>\n\n\n\n<p>\u8fd9\u6837\uff0c\u5373\u4f7f\u5e94\u7528\u5b95\u673a\u6216\u91cd\u542f\uff0c\u6211\u4eec\u7684\u4efb\u52a1\u8fdb\u5ea6\u4f9d\u7136\u53ef\u4ee5\u5224\u65ad\u662f\u5426\u5df2\u7ecf\u6267\u884c\u8fc7\uff0c\u6267\u884c\u8fc7\u4ec5\u6267\u884c\u81ea\u589e\u5185\u5b58\u4e2d\u7684\u8bfb\u53d6\u8fdb\u5ea6\u5373\u53ef\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u91cd\u6784\u6d88\u606f\u961f\u5217\u6267\u884c<\/h2>\n\n\n\n<p>\u5982\u4e0a\u6240\u8ff0\uff0c\u5982\u679c\u6d88\u606f\u6d88\u8d39\u65f6\u95f4\u8fc7\u957f\uff0c\u53ef\u80fd\u4f1a\u5bfc\u81f4\u610f\u5916\u95ee\u9898\uff0c\u4f8b\u5982 Broker \u53ef\u80fd\u4f1a\u8ba4\u4e3a\u6d88\u606f\u6d88\u8d39\u8d85\u65f6\u5e76\u91cd\u65b0\u6295\u9012\u3002\u4e3a\u4e86\u89e3\u51b3\u8fd9\u4e00\u95ee\u9898\uff0c\u6211\u4eec\u5c06\u6d88\u8d39\u8005\u62c6\u5206\u4e3a\u4e24\u4e2a\u90e8\u5206\uff1a\u4e00\u4e2a\u8d1f\u8d23\u89e3\u6790\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u7684 Excel \u6a21\u677f\uff0c\u53e6\u4e00\u4e2a\u8d1f\u8d23\u5b9e\u9645\u7684\u4f18\u60e0\u5238\u4efb\u52a1\u5206\u53d1\u3002<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>CouponTaskExecuteConsumer\uff1a\u5728\u6267\u884c\u4f18\u60e0\u5238\u5206\u53d1\u65f6\uff0c\u6211\u4eec\u540c\u65f6\u8fdb\u884c Excel \u6a21\u677f\u89e3\u6790\u548c\u524d\u7f6e\u6821\u9a8c\uff0c\u5305\u62ec\u6570\u636e\u683c\u5f0f\u7684\u6b63\u786e\u6027\u68c0\u67e5\u4ee5\u53ca\u5f53\u524d\u4f18\u60e0\u5238\u6a21\u677f\u7684\u5e93\u5b58\u60c5\u51b5\u3002<\/li>\n\n\n\n<li>CouponExecuteDistributionConsumer\uff1a\u5c06\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\u8d26\u6237\uff0c\u5305\u62ec\u6570\u636e\u5e93\u548c\u7f13\u5b58\u7b49\u591a\u4e2a\u5b58\u50a8\u4ecb\u8d28\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u901a\u8fc7\u8fd9\u79cd\u62c6\u5206\u673a\u5236\uff0c\u6211\u4eec\u53ef\u4ee5\u6700\u5927\u7a0b\u5ea6\u5730\u7f29\u77ed\u5355\u4e2a\u6d88\u606f\u961f\u5217\u6d88\u8d39\u8005\u7684\u6267\u884c\u65f6\u95f4\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u6279\u91cf\u4fdd\u5b58\u7528\u6237\u4f18\u60e0\u5238<\/h2>\n\n\n\n<p>\u4e4b\u524d\u5904\u7406\u5927\u7ea6 5000 \u6761 Excel \u5206\u53d1\u8bb0\u5f55\u9700\u8981\u7ea6 1 \u5206\u949f\u3002\u5982\u679c\u9700\u8981\u5904\u7406 100 \u4e07\u6761\u8bb0\u5f55\uff0c\u9884\u8ba1\u8017\u65f6\u5c06\u8fbe\u5230 200 \u5206\u949f\uff0c\u8fd9\u663e\u7136\u662f\u4e0d\u53ef\u63a5\u53d7\u7684\u3002\u6211\u4eec\u5206\u6790\u53d1\u73b0\uff0c\u5927\u91cf\u7684 Redis \u548c MySQL \u64cd\u4f5c\u9700\u8981\u9891\u7e41\u7684\u7f51\u7edc I\/O\uff0c\u800c\u7f51\u7edc\u64cd\u4f5c\u76f8\u6bd4\u4e8e CPU \u5185\u90e8\u5904\u7406\u8981\u6162\u5f97\u591a\u3002\u56e0\u6b64\uff0c\u6211\u4eec\u9700\u8981\u5c06\u6240\u6709\u6d89\u53ca\u7f51\u7edc I\/O \u7684\u64cd\u4f5c\u6539\u4e3a\u6279\u91cf\u5904\u7406\u3002<\/p>\n\n\n\n<p>\u6279\u91cf\u5904\u7406\u5206\u4e3a\u4e24\u4e2a\u6b65\u9aa4\uff1a\u6279\u91cf\u4fdd\u5b58\u5230\u6570\u636e\u5e93\u548c\u6279\u91cf\u4fdd\u5b58\u5230\u7f13\u5b58\u3002<\/p>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1\">1. \u6279\u91cf\u4fdd\u5b58\u7528\u6237\u5206\u53d1\u8bb0\u5f55<\/h3>\n\n\n\n<p>\u5728\u6267\u884c\u5206\u53d1\u4e4b\u524d\uff0c\u6211\u4eec\u9700\u8981\u9996\u5148\u9a8c\u8bc1\u4f18\u60e0\u5238\u6a21\u677f\u7684\u5e93\u5b58\u662f\u5426\u5145\u8db3\u3002\u5982\u679c\u5e93\u5b58\u4e0d\u8db3\uff0c\u5219\u65e0\u6cd5\u6267\u884c\u5206\u53d1\u3002\u6b63\u5982\u4e4b\u524d\u63d0\u5230\u7684\uff0c\u6211\u4eec\u9700\u8981\u5c06\u6d89\u53ca\u7f51\u7edc I\/O \u7684\u8bf7\u6c42\u8fdb\u884c\u6279\u91cf\u5904\u7406\u3002\u540c\u65f6\uff0c\u6211\u4eec\u5c06\u6d88\u606f\u53d1\u9001\u4efb\u52a1\u62c6\u5206\u7ed9\u4e86\u4e24\u4e2a\u6d88\u8d39\u8005\uff0c\u5176\u4e2d\u5b9e\u9645\u7684\u53d1\u9001\u64cd\u4f5c\u5728 <code>CouponExecuteDistributionConsumer<\/code> \u4e2d\u5b8c\u6210\u3002\u8fd9\u5c31\u5e26\u6765\u4e86\u4e00\u4e2a\u95ee\u9898\uff1a\u5982\u679c\u4e0d\u6ee1\u8db3\u6279\u91cf\u53d1\u9001\u6761\u4ef6\uff0c\u6570\u636e\u5e94\u8be5\u6682\u5b58\u5728\u54ea\u91cc\uff1f\u5bf9\u6b64\uff0c\u6211\u4eec\u9009\u62e9\u5c06\u6570\u636e\u6682\u5b58\u5230 Redis \u7684 Set \u7f13\u5b58\u4e2d\uff0c\u5f85\u6ee1\u8db3\u6279\u91cf\u6761\u4ef6\u65f6\u518d\u4ece Redis Set \u4e2d\u63d0\u53d6\u5e76\u6267\u884c\u6279\u91cf\u4fdd\u5b58\u3002<\/p>\n\n\n\n<p><code>ReadExcelDistributionListener#invoke<\/code> \u6838\u5fc3\u903b\u8f91\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> \/\/ \u6267\u884c LUA \u811a\u672c\u8fdb\u884c\u6263\u51cf\u5e93\u5b58\u4ee5\u53ca\u589e\u52a0 Redis \u7528\u6237\u9886\u5238\u8bb0\u5f55\n String couponTemplateKey = String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, couponTemplateDO.getId());\n String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, couponTaskId);\n Map&lt;Object, Object&gt; userRowNumMap = MapUtil.builder()\n         .put(\"userId\", data.getUserId())\n         .put(\"rowNum\", rowCount + 1)\n         .build();\n Long combinedFiled = stringRedisTemplate.execute(buildLuaScript, ListUtil.of(couponTemplateKey, batchUserSetKey), JSON.toJSONString(userRowNumMap));\n \u200b\n \/\/ firstField \u4e3a false \u8bf4\u660e\u4f18\u60e0\u5238\u5df2\u7ecf\u6ca1\u6709\u5e93\u5b58\u4e86\n boolean firstField = StockDecrementReturnCombinedUtil.extractFirstField(combinedFiled);\n if (!firstField) {\n     \/\/ \u540c\u6b65\u5f53\u524d\u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n     stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n     ++rowCount;\n \u200b\n     \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n     Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n             .put(\"rowNum\", rowCount)\n             .put(\"cause\", \"\u4f18\u60e0\u5238\u6a21\u677f\u65e0\u5e93\u5b58\")\n             .build();\n     CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n             .batchId(couponTaskDO.getBatchId())\n             .jsonObject(JSON.toJSONString(objectMap, SerializerFeature.WriteMapNullValue))\n             .build();\n     couponTaskFailMapper.insert(couponTaskFailDO);\n     return;\n }\n \u200b\n \/\/ \u83b7\u53d6\u7528\u6237\u9886\u5238\u96c6\u5408\u957f\u5ea6\n int batchUserSetSize = StockDecrementReturnCombinedUtil.extractSecondField(combinedFiled.intValue());<\/code><\/pre>\n\n\n\n<p>\u53ef\u80fd\u5f88\u591a\u540c\u5b66\u4e0d\u6e05\u695a\uff0c\u4e3a\u4ec0\u4e48\u8981\u628a Lua \u8fd4\u56de\u811a\u672c\u4e2d\u7684\u5b57\u6bb5\u901a\u8fc7\u5de5\u5177\u7c7b\u62c6\u5206\u4e3a\u4e24\u4e2a\u503c\uff0c\u5148\u770b\u770b\u4e0b\u6587\u7684\u89e3\u91ca\u3002<\/p>\n\n\n\n<p>\u9996\u5148\u7f16\u5199\u4e00\u4e2a Lua \u811a\u672c\uff0c\u7528\u4e8e\u9a8c\u8bc1\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u662f\u5426\u5145\u8db3\u3002\u5982\u679c\u5e93\u5b58\u5145\u8db3\uff0c\u5219\u5c06\u6570\u636e\u4fdd\u5b58\u5230 Redis \u7684 Set \u7ed3\u6784\u4e2d\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> -- \u5b9a\u4e49\u6700\u5927\u503c\u548c\u4f4d\u6570\n local SECOND_FIELD_BITS = 13\n \u200b\n -- \u5c06\u4e24\u4e2a\u5b57\u6bb5\u7ec4\u5408\u6210\u4e00\u4e2aint\n local function combineFields(firstField, secondField)\n     local firstFieldValue = firstField and 1 or 0\n     return (firstFieldValue * 2 ^ SECOND_FIELD_BITS) + secondField\n end\n \u200b\n -- Lua\u811a\u672c\u5f00\u59cb\n local key = KEYS&#91;1] -- Redis Key\n local userSetKey = KEYS&#91;2] -- \u7528\u6237\u9886\u5238 Set \u7684 Key\n local userIdAndRowNum = ARGV&#91;1] -- \u7528\u6237 ID \u548c Excel \u6240\u5728\u884c\u6570\n \u200b\n -- \u83b7\u53d6\u5e93\u5b58\n local stock = tonumber(redis.call('HGET', key, 'stock'))\n \u200b\n -- \u68c0\u67e5\u5e93\u5b58\u662f\u5426\u5927\u4e8e0\n if stock == nil or stock &lt;= 0 then\n     return combineFields(false, redis.call('SCARD', userSetKey))\n end\n \u200b\n -- \u81ea\u51cf\u5e93\u5b58\n redis.call('HINCRBY', key, 'stock', -1)\n \u200b\n -- \u6dfb\u52a0\u7528\u6237\u5230\u9886\u5238\u96c6\u5408\n redis.call('SADD', userSetKey, userIdAndRowNum)\n \u200b\n -- \u83b7\u53d6\u7528\u6237\u9886\u5238\u96c6\u5408\u7684\u957f\u5ea6\n local userSetLength = redis.call('SCARD', userSetKey)\n \u200b\n -- \u8fd4\u56de\u7ed3\u679c\n return combineFields(true, userSetLength)<\/code><\/pre>\n\n\n\n<p>\u5927\u5bb6\u770b\u5230\u8fd9\u4e2a\u53ef\u80fd\u6709\u70b9\u8499\u5708\uff0c\u7b80\u5355\u89e3\u91ca\u4e0b\u3002\u8fd9\u4e2a LUA \u811a\u672c\u4e3b\u8981\u505a\u4e24\u4ef6\u4e8b\u60c5\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u5224\u65ad\u5e93\u5b58\u662f\u5426\u5927\u4e8e 0\uff0c\u5982\u679c\u5927\u4e8e\u7684\u8bdd\uff0c\u81ea\u51cf\u5e93\u5b58\u3002\u5982\u679c\u5e93\u5b58\u4e3a\u7a7a\u6216\u5c0f\u4e8e\u7b49\u4e8e 0 \u5219\u8fd4\u56de\u5931\u8d25\uff1b<\/li>\n\n\n\n<li>2. \u6263\u51cf\u5e93\u5b58\u6210\u529f\u540e\uff0c\u6211\u4eec\u9700\u8981\u5c06\u7528\u6237\u9886\u5238\u8bb0\u5f55\u4fdd\u5b58\u5230 Redis Set \u7ed3\u6784\u4e2d\uff0c\u65b9\u4fbf\u540e\u9762\u6279\u91cf\u8fdb\u884c\u5206\u53d1\u4fdd\u5b58\u6570\u636e\u5e93\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u8fd9\u4e2a Lua \u811a\u672c\u7684\u547d\u4ee4\u867d\u7136\u7b80\u5355\uff0c\u4f46\u901a\u8fc7<strong>\u4f4d\u79fb\u64cd\u4f5c\u5de7\u5999\u63d0\u5347\u4e86\u8fd4\u56de\u6570\u636e\u7684\u6027\u80fd<\/strong> \u3002\u901a\u5e38\u60c5\u51b5\u4e0b\uff0c\u6211\u4eec\u9700\u8981\u8fd4\u56de\u4e24\u4e2a\u6570\u636e\uff1a\u5e93\u5b58\u662f\u5426\u5145\u8db3\uff0c\u4ee5\u53ca\u7528\u6237\u9886\u5238 Set \u7684\u957f\u5ea6\uff08\u5f53\u8fbe\u5230\u6307\u5b9a\u957f\u5ea6\u65f6\u5f00\u59cb\u6267\u884c\u6279\u91cf\u5206\u53d1\uff09\u3002\u5982\u679c\u6309\u7167\u5e38\u89c4\u65b9\u5f0f\u8fd4\u56de\uff0c\u6bd4\u5982\u8fd4\u56de <code>false,0-5000<\/code>\uff0c\u5c31\u9700\u8981\u5728 Java \u670d\u52a1\u4e2d\u8fdb\u884c\u5b57\u7b26\u4e32\u5206\u5272\u3002\u76f8\u6bd4\u4e4b\u4e0b\uff0c\u4f7f\u7528\u4e8c\u8fdb\u5236\u4f4d\u79fb\u64cd\u4f5c\u53ef\u4ee5\u5927\u5e45\u63d0\u5347\u6027\u80fd\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u9996\u5148\uff0c\u5e93\u5b58\u5982\u679c\u8bf4\u5c0f\u4e8e\u7b49\u4e8e0\u7684\u8bdd\uff0c\u90a3\u5c31\u8bc1\u660e\u6211\u4eec\u7684\u5e93\u5b58\u4e0d\u591f\u4e86\uff0c\u6211\u4eec\u8981\u8fd4\u56de\u4e00\u4e2a\u5931\u8d25\uff0c\u4ee5\u53ca\u6211\u4eec\u5f53\u524d\u7f13\u51b2\u961f\u5217\u91cc\u9762\u5b83\u5df2\u7ecf\u6709\u4e86\u591a\u5c11\u884c\u6570\u2014\u2014\u63a5\u4e0b\u6765\u7684\u4e1a\u52a1\u5f53\u4e2d\u4f1a\u7528\u5230\u8fd9\u4e2a\u6570\u636e\u3002\u8fd9\u91cc\u5148\u5ffd\u7565\u6389\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u5982\u679c\u8bf4\u5b83\u7684\u5e93\u5b58\u662f\u5145\u8db3\u7684\uff0c\u90a3\u4e48\u6211\u4eec\u4f1a\u81ea\u51cf\u5b83\u7684\u5e93\u5b58\uff0c\u7136\u540e\u5c06\u5b83\u7684\u7528\u6237\u4fe1\u606f\u7ed9\u5b83\u6dfb\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u9886\u5238\u96c6\u5408\u91cc\u9762\u53bb\u3002\u5176\u5b9e\u6211\u4eec\u5c31\u662f\u7528\u4e86\u4e00\u4e2a Set \u7684\u6570\u636e\u7ed3\u6784\u53bb\u4fdd\u5b58\u4e86\u8bb0\u5f55\u3002\u540c\u65f6\uff0c\u6211\u4eec\u4e5f\u8981\u8fd4\u56de\u7528\u6237\u9886\u5238\u96c6\u5408\u7684\u4e00\u4e2a\u957f\u5ea6\u3002<\/p>\n\n\n\n<p>\u8fd9\u4e2a\u957f\u5ea6\u662f\u7528\u6765\u505a\u4ec0\u4e48\u7684\uff1f\u8fd9\u4e2a\u5c31\u662f\u7528\u6765\u521a\u624d\u6211\u4eec\u8bb2\u7684\u201c\u6279\u6b21\u201d\u7684\u6982\u5ff5\u3002\u5982\u679c\u8bf4\u5b83\u8fbe\u5230\u4e86\u4e00\u4e2a\u6211\u4eec\u6279\u6b21\u7684\u8981\u6c42\uff0c\u5c31\u8981\u7ed9\u5b83\u8fdb\u884c\u4e00\u6b21\u5904\u7406\uff0c\u5bf9\u5427\uff1f<\/p>\n\n\n\n<p>\u7136\u540e\u8fd9\u91cc\u8fd4\u56de\u7ed3\u679c\u7684\u65f6\u5019\uff0c\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\u5b83\u4f1a\u53bb\u8c03\u8fd9\u4e2a\u51fd\u6570\u3002\u8fd9\u4e2a\u51fd\u6570\u5176\u5b9e\u5c31\u662f\u4e00\u4e2a\u4f5c\u7528\uff1a\u90a3\u5c31\u662f\u5b83\u4f1a\u628a\u4e24\u4e2a\u5b57\u6bb5\u7ec4\u5408\u6210\u4e00\u4e2a int \u53bb\u8fdb\u884c\u8fd4\u56de\u3002\u7136\u540e\u6211\u4eec\u5728\u4e1a\u52a1\u91cc\u9762\u901a\u8fc7\u4f4d\u8fd0\u7b97\u7684\u5f62\u5f0f\uff0c\u628a\u8fd9\u4e2a int \u518d\u89e3\u6790\u6210\u4e24\u4e2a\u5b57\u6bb5\u3002<\/p>\n\n\n\n<p>\u5927\u5bb6\u4e00\u542c\u8fd9\u4e2a\u903b\u8f91\u64cd\u4f5c\u53ef\u80fd\u4f1a\u6709\u70b9\u61f5\uff1a\u4e3a\u4ec0\u4e48\u4f1a\u8fd9\u4e48\u53bb\u5199\uff1f<\/p>\n\n\n\n<p>\u56e0\u4e3a\u5982\u679c\u8bf4\u6211\u4eec\u4e0d\u5bf9\u5b83\u8fdb\u884c\u8fd9\u79cd\u4f4d\u8fd0\u7b97\u7684\u7ec4\u5408\uff0c\u90a3\u4e48\u6211\u4eec\u662f\u4e0d\u662f\u9700\u8981\u7c7b\u4f3c\u4e8e\u8fd9\u79cd\u7ed3\u6784\uff1f\u7b2c\u4e00\u4e2a\u662f\u201c\u662f\u5426\u5e93\u5b58\u6263\u51cf\u6210\u529f\u201d\u7684\u5b57\u6bb5\u3002\u5982\u679c\u8bf4\u6211\u4eec\u4e0d\u8fdb\u884c\u4f4d\u8fd0\u7b97\u62fc\u63a5\uff0c\u6211\u4eec\u4e00\u5b9a\u662f\u8fd4\u56de\u4e24\u4e2a\u503c\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u8fd9\u6837\u7684\u8bdd\uff0c\u4f60\u662f\u4e0d\u662f\u5230\u4e86\u4e1a\u52a1\u4ee3\u7801\u91cc\u9762\uff0c\u9700\u8981\u53bb\u8fdb\u884c\u4e00\u4e2a\u5206\u5272\uff1f<\/p>\n\n\n\n<p>\u5728\u8fd9\u91cc\u8ddf\u5927\u5bb6\u8bb2\u4e00\u4e0b\uff1a\u5206\u5272\u7684\u6027\u80fd\u662f\u8fdc\u4e0d\u5982\u4f4d\u8fd0\u7b97\u7684\u3002\u597d\u5427\uff1f\u6240\u4ee5\u8bf4\u8fd9\u91cc\u9762\u662f\u4e00\u4e2a\u201c\u82b1\u6d3b\u201d\u3002\u5927\u5bb6\u80fd\u591f\u5728\u4ee3\u7801\u91cc\u9762\u770b\u5230\u4e00\u4e9b\u6838\u5fc3\u4ee3\u7801\u8fd9\u4e48\u5199\uff0c\u4e0d\u5efa\u8bae\u5927\u5bb6\u5728\u666e\u901a\u4e1a\u52a1\u91cc\u53bb\u7528\uff0c\u907f\u514d\u6709\u201c\u79c0\u201d\u7684\u5acc\u7591\u3002\u4f46\u4f60\u8981\u77e5\u9053\uff1a\u8fd9\u6837\u7684\u64cd\u4f5c\u5728\u4e00\u4e9b\u6838\u5fc3\u4ee3\u7801\u91cc\u9762\u53ef\u4ee5\u8fd9\u6837\u5199\uff0c\u5e2e\u52a9\u5927\u5bb6\u53bb\u6269\u5c55\u4e00\u4e0b\u601d\u7ef4\u3002<\/p>\n\n\n\n<p>\u53e6\u5916\uff0c\u5982\u679c\u7ec6\u5fc3\u7684\u5c0f\u4f19\u4f34\uff0c\u5305\u62ec\u770b\u8fc7\u4e00\u4e9b\u7ebf\u7a0b\u6c60\u6e90\u7801\u7684\u5c0f\u4f19\u4f34\uff0c\u53ef\u80fd\u5c31\u6709\u60f3\u6cd5\u4e86\u2014\u2014\u5bf9\u5427\uff1f\u7ebf\u7a0b\u6c60\u7684\u72b6\u6001\u5176\u5b9e\u4e5f\u662f\u8fd9\u4e48\u53bb\u505a\u7684\uff0c\u4e5f\u662f\u901a\u8fc7\u4e00\u4e2a\u5b57\u6bb5\u8868\u793a\u4e86\u5b83\u7684\u591a\u4e2a\u5c5e\u6027\u3002<\/p>\n<\/blockquote>\n\n\n\n<p>\u8fd4\u56de\u7684\u65f6\u5019\uff0c\u53ef\u4ee5\u770b\u5230\u5b83\u4f1a\u8fd4\u56de\u4e00\u4e2a\u6570\u503c\u3002\u7136\u540e\u901a\u8fc7\u6570\u503c\uff0c\u6211\u4eec\u5c01\u88c5\u4e86\u4e00\u4e2a\u5de5\u5177\u7c7b\uff0c\u5e2e\u52a9\u8fd9\u4e2a\u6570\u503c\u53bb\u8fdb\u884c\u4e00\u4e2a\u89e3\u6790\u3002\u53ef\u4ee5\u770b\u5230\u5b83\u8fd4\u56de\u4e86\u4e00\u4e2a 8193\uff0c\u8fd9\u4e2a\u6570\u636e\u5bf9\u54b1\u6765\u8bf4\u6ca1\u6709\u4efb\u4f55\u610f\u4e49\uff0c\u6216\u8005\u8bf4\u6ca1\u6709\u4efb\u4f55\u53ef\u8bfb\u6027\u3002\u6240\u4ee5\u6211\u4eec\u9700\u8981\u901a\u8fc7\u5de5\u5177\u7c7b\u7684\u5f62\u5f0f\u53bb\u89e3\u6790\u3002\u5c31\u662f\u4e0b\u9762<\/p>\n\n\n\n<p>\u6570\u636e\u8fd4\u56de\u540e\uff0c\u6211\u4eec\u901a\u8fc7\u81ea\u5b9a\u4e49\u7684\u4e00\u4e2a\u5de5\u5177\u7c7b\uff0c\u89e3\u6790\u8fd4\u56de\u6570\u636e\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> public class StockDecrementReturnCombinedUtil {\n \u200b\n     \/**\n      * 2^13 &gt; 5000, \u6240\u4ee5\u7528 13 \u4f4d\u6765\u8868\u793a\u7b2c\u4e8c\u4e2a\u5b57\u6bb5\n      *\/\n     private static final int SECOND_FIELD_BITS = 13;\n \u200b\n     \/**\n      * \u5c06\u4e24\u4e2a\u5b57\u6bb5\u7ec4\u5408\u6210\u4e00\u4e2aint\n      *\/\n     public static int combineFields(boolean decrementFlag, int userRecord) {\n         return (decrementFlag ? 1 : 0) &lt;&lt; SECOND_FIELD_BITS | userRecord;\n     }\n \u200b\n     \/**\n      * \u4ece\u7ec4\u5408\u7684int\u4e2d\u63d0\u53d6\u7b2c\u4e00\u4e2a\u5b57\u6bb5\uff080\u62161\uff09\n      *\/\n     public static boolean extractFirstField(long combined) {\n         return (combined &gt;&gt; SECOND_FIELD_BITS) != 0;\n     }\n \u200b\n     \/**\n      * \u4ece\u7ec4\u5408\u7684int\u4e2d\u63d0\u53d6\u7b2c\u4e8c\u4e2a\u5b57\u6bb5\uff081\u52305000\u4e4b\u95f4\u7684\u6570\u5b57\uff09\n      *\/\n     public static int extractSecondField(int combined) {\n         return combined &amp; ((1 &lt;&lt; SECOND_FIELD_BITS) - 1);\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u7ecf\u8fc7\u8fd9\u4e2a\u5de5\u5177\u7c7b\uff0c\u5c31\u53ef\u4ee5\u5b9e\u73b0\u6211\u4eec\u4e0a\u9762\u7684\u4e00\u4e2a\u5b57\u6bb5\u5b8c\u6210\u4e24\u4e2a\u503c\u7684\u5b58\u50a8\u3002\u53e6\u5916\uff0c\u6211\u4eec\u901a\u8fc7\u4e00\u4e2a\u5355\u5143\u6d4b\u8bd5\u80fd\u591f\u5f88\u76f4\u89c2\u7684\u770b\u5230\u4e24\u79cd\u65b9\u5f0f\u7684\u6027\u80fd\u5dee\u8ddd\uff0c\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> @Slf4j\n public final class StockDecrementReturnCombinedUtilTests {\n \u200b\n     @Test\n     public void stockDecrementReturnCombinedUtilTest() {\n         boolean firstField = true;\n         int secondField = 5000;\n \u200b\n         int combined = StockDecrementReturnCombinedUtil.combineFields(firstField, secondField);\n \u200b\n         long startTime = System.currentTimeMillis();\n         for (int i = 0; i &lt; 100000; i++) {\n             StockDecrementReturnCombinedUtil.extractFirstField(combined);\n             StockDecrementReturnCombinedUtil.extractSecondField(combined);\n         }\n         long endTime = System.currentTimeMillis();\n \u200b\n \u200b\n         long startTime2 = System.currentTimeMillis();\n         String str = \"1,1234\";\n         for (int i = 0; i &lt; 100000; i++) {\n             StrUtil.split(str, \",\");\n         }\n         long endTime2 = System.currentTimeMillis();\n \u200b\n         log.info(\"\u4f4d\u79fb\u7a0b\u5e8f\u6267\u884c\u65f6\u95f4\uff1a{}\", endTime - startTime);\n         log.info(\"split\u7a0b\u5e8f\u6267\u884c\u65f6\u95f4\uff1a{}\", endTime2 - startTime2);\n         \/**\n          * \u4f4d\u79fb\u7a0b\u5e8f\u6267\u884c\u65f6\u95f4\uff1a2\n          * split\u7a0b\u5e8f\u6267\u884c\u65f6\u95f4\uff1a40\n          *\/\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u8fd0\u884c 10 \u4e07\u6b21\u65f6\uff0c\u6027\u80fd\u5dee\u8ddd\u901a\u5e38\u4f1a\u975e\u5e38\u660e\u663e\uff0c\u53ef\u80fd\u8fbe\u5230 20 \u500d\u5de6\u53f3\u3002\u7136\u800c\uff0c\u8fd9\u79cd\u4f18\u5316\u5728\u6211\u4eec\u7684\u5e94\u7528\u7cfb\u7edf\u4e2d\u5b9e\u9645\u610f\u4e49\u4e0d\u5927\uff0c\u53ea\u662f\u8ba9\u5927\u5bb6\u4e86\u89e3\u8fd9\u79cd\u65b9\u6cd5\u7684\u5b58\u5728\u3002\u6bd5\u7adf\u5728\u6267\u884c\u65f6\u95f4\u4e0a\uff0c1 \u6beb\u79d2\u4e0e 2 \u6beb\u79d2\u7684\u5dee\u8ddd\u5f88\u96be\u5bdf\u89c9\u3002\u8fd9\u79cd\u4f18\u5316\u601d\u8def\u6765\u6e90\u4e8e\u7ebf\u7a0b\u6c60\u7684 <code>ctl<\/code> \u5b57\u6bb5\uff0c\u5176\u6838\u5fc3\u601d\u60f3\u4e0e\u6211\u4eec\u8fd9\u91cc\u7684\u4e00\u81f4\uff0c\u5373\u901a\u8fc7\u4f4d\u79fb\u64cd\u4f5c\u8ba9\u4e00\u4e2a\u5b57\u6bb5\u8868\u793a\u591a\u79cd\u7528\u9014\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> public class ThreadPoolExecutor extends AbstractExecutorService {\n     \/\/ ......\n     private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0)); \n     \/\/ ......\n }<\/code><\/pre>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5728\u8fd9\u91cc\u7684\u8bdd\uff0c\u6211\u4eec\u662f\u9700\u8981\u7b2c\u4e00\u4e2a\u5b57\u6bb5\uff1a\u6211\u4eec\u7684\u4f18\u60e0\u5238\u662f\u5426\u8fd8\u6709\u5e93\u5b58\u3002<\/p>\n\n\n\n<p>\u6211\u4eec\u7b2c\u4e00\u4e2a\u5b57\u6bb5\u600e\u4e48\u89e3\u6790\uff1f\u9996\u5148\u901a\u8fc7\u4f4d\u8fd0\u7b97\uff0c\u6211\u4eec\u53ef\u4ee5\u770b\u5230\u8fd9\u4e2a\u65b9\u6cd5\uff0c\u5b83\u7684\u4f5c\u7528\u5176\u5b9e\u5c31\u662f\u4ece\u6211\u4eec\u7684 int \u91cc\u9762\u53bb\u63d0\u53d6\u51fa\u6765\u7b2c\u4e00\u4e2a\u5b57\u6bb5\uff080 \u6216 1\uff09\uff0c\u662f\u901a\u8fc7\u4f4d\u79fb\u8fd0\u7b97\u7684\u5f62\u5f0f\u63d0\u51fa\u6765\u7684\u3002\u63d0\u51fa\u6765\u4e4b\u540e\uff0c\u6211\u4eec\u5c31\u53ef\u4ee5\u77e5\u9053\u5b83\u662f\u4e00\u4e2a true \u6216 false\u3002<\/p>\n\n\n\n<p>\u5982\u679c\u8bf4\u5b83\u7b49\u4e8e false\uff0c\u5c31\u8bc1\u660e\u6211\u4eec\u4f18\u60e0\u5238\u91cc\u9762\u5df2\u7ecf\u6ca1\u6709\u5e93\u5b58\u4e86\u3002\u8fd9\u4e2a\u65f6\u5019\u6211\u4eec\u4f1a\u6267\u884c\u4e24\u4e2a\u903b\u8f91\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u6b65\uff0c\u628a\u6211\u4eec\u5f53\u524d\u7684\u4e00\u4e2a\u8fdb\u5ea6\u653e\u5230\u7f13\u5b58\u91cc\u9762\u53bb\uff1b<\/li>\n\n\n\n<li>\u7136\u540e\uff0c\u6211\u4eec\u53bb\u628a\u5b83\u5bf9\u5e94\u7684\u4e00\u4e2a\u884c\u6570\u4ee5\u53ca\u5bf9\u5e94\u7684\u4e00\u4e2a\u5931\u8d25\u539f\u56e0\uff0c\u65b0\u589e\u5230\u6211\u4eec\u521a\u624d\u8bb2\u5230\u7684\u4efb\u52a1\u5931\u8d25\u7684\u8bb0\u5f55\u8868\u91cc\u9762\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5230\u8fd9\u91cc\u5c31\u7ed3\u675f\u4e86\u3002\u7136\u540e\u6211\u4eec\u53ef\u4ee5\u770b\u5230\uff0c\u5176\u5b9e\u73b0\u5728\u80af\u5b9a\u662f\u6709\u8bb0\u5f55\u7684\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u7b2c\u4e8c\u4e2a\u5b57\u6bb5\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u7684\u4f18\u60e0\u5238\u7f13\u51b2\u961f\u5217\u7684\u4e00\u4e2a\u957f\u5ea6\u2014\u2014\u662f 1~5000 \u4e4b\u95f4\u7684\u4e00\u4e2a\u6570\u5b57\u3002<\/p>\n\n\n\n<p>\u53ef\u80fd\u6709\u540c\u5b66\u4f1a\u7ea0\u7ed3\uff1a\u201c\u9a6c\u54e5\uff0c\u6211\u7b2c\u4e8c\u4e2a\u957f\u5ea6\u7684\u8bdd\u662f\u591a\u5927\u6bd4\u8f83\u5408\u9002\uff1f\u201d<\/p>\n\n\n\n<p>\u6211\u4eec\u5728\u8fd9\u91cc\u7528\u7684\u662f 2 \u7684 13 \u6b21\u5e42\uff088192\uff09\uff0c\u5bf9\u5427\uff1f\u4ee3\u8868\u7684\u662f 8192\uff0c\u80af\u5b9a\u662f\u5927\u4e8e 5000 \u7684\u3002\u6211\u4eec\u7559\u8fd9\u4e2a\u957f\u5ea6\u8ba9\u5b83\u6765\u89e3\u6790\uff0c\u7136\u540e\u6211\u4eec\u7ed9\u5b83\u89e3\u6790\u51fa\u6765\u4e4b\u540e\uff0c\u5c31\u80fd\u591f\u83b7\u53d6\u5230\u5b83\u7b2c\u4e8c\u4e2a\u6570\u636e\u3002<\/p>\n\n\n\n<p>\u7b2c\u4e8c\u4e2a\u6570\u636e\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\uff0c\u5728\u8fd9\u91cc\u7684\u8bdd\u662f 1\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u56e0\u4e3a\u6211\u4eec\u521a\u6267\u884c\u7b2c\u4e00\u6b21\u3002\u5982\u679c\u5b83\u662f\u4e00\u7684\u8bdd\uff0c\u53ef\u4ee5\u770b\u5230\u5b83\u662f\u4e0d\u6ee1\u8db3\u6211\u4eec\u7684\u6279\u6b21\u7684\uff08\u4e00\u4e2a\u6279\u6b21\u662f 5000\uff09\u3002\u5982\u679c\u4e0d\u6ee1\u8db3\u7684\u8bdd\uff0c\u548c\u521a\u624d\u7684\u903b\u8f91\u4e00\u6837\uff0c\u6211\u4eec\u7ed9\u4ed6\u7edf\u8ba1\u4e00\u4e0b\u5b83\u5bf9\u5e94\u7684\u8fdb\u5ea6\u5c31 OK \u4e86\uff0c\u7136\u540e\u6211\u4eec\u628a\u5bf9\u5e94\u7684\u903b\u8f91\u52a0\u5230\u8fd9\u91cc\u3002<\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u8fbe\u5230\u6761\u4ef6\u6267\u884c\u6d88\u606f\u961f\u5217\u5206\u53d1\u6d88\u8d39\u8005<\/h3>\n\n\n\n<p>\u6211\u4eec\u8bbe\u5b9a\u6bcf\u8bfb\u53d6 5000 \u6761\u8bb0\u5f55\u540e\u6267\u884c\u4e00\u6b21\u6279\u91cf\u64cd\u4f5c\uff0c\u5982\u679c\u4e0d\u8db3 5000 \u6761\uff0c\u5219\u53ea\u540c\u6b65\u5f53\u524d\u7684 Excel \u8bfb\u53d6\u8fdb\u5ea6\u3002\u5f53\u8bfb\u53d6\u8fdb\u5ea6\u8fbe\u5230 5000 \u6761\u65f6\uff0c\u5c06\u53d1\u9001\u6d88\u606f\u961f\u5217\u901a\u77e5\u5177\u4f53\u7684\u4f18\u60e0\u5238\u5206\u53d1\u6d88\u8d39\u8005\u8fdb\u884c\u5904\u7406\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> private final static int BATCH_USER_COUPON_SIZE = 5000;\n \u200b\n \/\/ \u83b7\u53d6\u7528\u6237\u9886\u5238\u96c6\u5408\u957f\u5ea6\n int batchUserSetSize = StockDecrementReturnCombinedUtil.extractSecondField(combinedFiled.intValue());\n \u200b\n \/\/ batchUserSetSize = BATCH_USER_COUPON_SIZE \u65f6\u53d1\u9001\u6d88\u606f\u6d88\u8d39\uff0c\u4e0d\u6ee1\u8db3\u6761\u4ef6\u4ec5\u8bb0\u5f55\u6267\u884c\u8fdb\u5ea6\u5373\u53ef\n if (batchUserSetSize &lt; BATCH_USER_COUPON_SIZE) {\n     \/\/ \u540c\u6b65\u5f53\u524d Excel \u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58\n     stringRedisTemplate.opsForValue().set(templateTaskExecuteProgressKey, String.valueOf(rowCount));\n     ++rowCount;\n     return;\n }\n \u200b\n CouponTemplateDistributionEvent couponTemplateDistributionEvent = CouponTemplateDistributionEvent.builder()\n         .couponTaskId(couponTaskId)\n         .shopNumber(couponTaskDO.getShopNumber())\n         .couponTemplateId(couponTemplateDO.getId())\n         .couponTaskBatchId(couponTaskDO.getBatchId())\n         .couponTemplateConsumeRule(couponTemplateDO.getConsumeRule())\n         .batchUserSetSize(batchUserSetSize)\n         .distributionEndFlag(Boolean.FALSE)\n         .build();\n couponExecuteDistributionProducer.sendMessage(couponTemplateDistributionEvent);<\/code><\/pre>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u65b0\u589e\u7528\u6237\u9886\u5238\u6570\u636e\u5e93<\/h3>\n\n\n\n<p>\u6267\u884c\u5206\u53d1\u903b\u8f91\u6d88\u606f\u961f\u5217\u6d88\u8d39\u8005\u65f6\uff0c\u4f1a\u6709\u4e24\u4e2a\u4e1a\u52a1\u7c7b\u578b\uff0c\u5206\u522b\u662f\u6ee1\u8db3\u6279\u91cf\u5904\u7406\u6570\u91cf 5000 \u7684\u6d88\u606f\uff0c\u4ee5\u53ca Excel \u8bfb\u53d6\u5230\u6700\u540e\u53ef\u80fd\u6ca1\u6ee1\u8db3 5000 \u6279\u91cf\u4e5f\u9700\u8981\u53d1\u9001\u6d88\u606f\u961f\u5217\u3002<\/p>\n\n\n\n<p>\u6279\u91cf\u8fbe\u5230 5000 \u6761\u6267\u884c\u7684\u6d88\u606f\u961f\u5217\u5982\u4e0a\u6240\u8ff0\uff0c\u7136\u540e\u7ed9\u5927\u5bb6\u770b\u4e0b EasyExcel \u8bfb\u53d6\u5b8c Excel \u540e\u6267\u884c\u7684\u540e\u7f6e\u903b\u8f91\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> @RequiredArgsConstructor\n public class ReadExcelDistributionListener extends AnalysisEventListener&lt;CouponTaskExcelObject&gt; {\n \u200b\n     \/\/ ......\n \u200b\n     @Override\n     public void doAfterAllAnalysed(AnalysisContext context) {\n         \/\/ \u53d1\u9001 Excel \u89e3\u6790\u5b8c\u6210\u6807\u8bc6\uff0c\u5373\u4f7f\u4e0d\u6ee1\u8db3\u6279\u91cf\u4fdd\u5b58\u7684\u6570\u91cf\u4e5f\u5f97\u4fdd\u5b58\u5230\u6570\u636e\u5e93\n         CouponTemplateDistributionEvent couponTemplateExecuteEvent = CouponTemplateDistributionEvent.builder()\n                 .distributionEndFlag(Boolean.TRUE) \/\/ \u8bbe\u7f6e\u89e3\u6790\u5b8c\u6210\u6807\u8bc6\n                 .shopNumber(couponTaskDO.getShopNumber())\n                 .couponTemplateId(couponTemplateDO.getId())\n                 .validEndTime(couponTemplateDO.getValidEndTime())\n                 .couponTemplateConsumeRule(couponTemplateDO.getConsumeRule())\n                 .couponTaskBatchId(couponTaskDO.getBatchId())\n                 .couponTaskId(couponTaskDO.getId())\n                 .build();\n         couponExecuteDistributionProducer.sendMessage(couponTemplateExecuteEvent);\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u6211\u4eec\u5148\u8bf4\u5176\u4e2d\u7684\u6267\u884c\u6279\u5904\u7406\u6d88\u606f\uff0c\u4ee3\u7801\u5982\u4e0b\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> @Component\n @RequiredArgsConstructor\n @RocketMQMessageListener(\n         topic = \"one-coupon_distribution-service_coupon-execute-distribution_topic${unique-name:}\",\n         consumerGroup = \"one-coupon_distribution-service_coupon-execute-distribution_cg${unique-name:}\"\n )\n @Slf4j(topic = \"CouponExecuteDistributionConsumer\")\n public class CouponExecuteDistributionConsumer implements RocketMQListener&lt;MessageWrapper&lt;CouponTemplateDistributionEvent&gt;&gt; {\n \u200b\n     private final UserCouponMapper userCouponMapper;\n     private final CouponTemplateMapper couponTemplateMapper;\n     private final CouponTaskMapper couponTaskMapper;\n     private final CouponTaskFailMapper couponTaskFailMapper;\n     private final StringRedisTemplate stringRedisTemplate;\n \u200b\n     @Lazy\n     @Autowired\n     private CouponExecuteDistributionConsumer couponExecuteDistributionConsumer;\n \u200b\n     private final static int BATCH_USER_COUPON_SIZE = 5000;\n     private static final String BATCH_SAVE_USER_COUPON_LUA_PATH = \"lua\/batch_user_coupon_list.lua\";\n \u200b\n     @Transactional(rollbackFor = Exception.class)\n     @Override\n     public void onMessage(MessageWrapper&lt;CouponTemplateDistributionEvent&gt; messageWrapper) {\n         \/\/ \u5f00\u5934\u6253\u5370\u65e5\u5fd7\uff0c\u5e73\u5e38\u53ef Debug \u770b\u4efb\u52a1\u53c2\u6570\uff0c\u7ebf\u4e0a\u53ef\u62a5\u5e73\u5b89\uff08\u6bd4\u5982\u6d88\u606f\u662f\u5426\u6d88\u8d39\uff0c\u91cd\u65b0\u6295\u9012\u65f6\u83b7\u53d6\u53c2\u6570\u7b49\uff09\n         log.info(\"&#91;\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u4efb\u52a1\u6267\u884c\u63a8\u9001@\u5206\u53d1\u5230\u7528\u6237\u8d26\u53f7 - \u6267\u884c\u6d88\u8d39\u903b\u8f91\uff0c\u6d88\u606f\u4f53\uff1a{}\", JSON.toJSONString(messageWrapper));\n \u200b\n         \/\/ \u5f53\u4fdd\u5b58\u7528\u6237\u4f18\u60e0\u5238\u96c6\u5408\u8fbe\u5230\u6279\u91cf\u4fdd\u5b58\u6570\u91cf\n         CouponTemplateDistributionEvent event = messageWrapper.getMessage();\n         if (!event.getDistributionEndFlag() &amp;&amp; event.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0) {\n             decrementCouponTemplateStockAndSaveUserCouponList(event);\n             return;\n         }\n       \n         \/\/ ......\n     }\n \u200b\n     @SneakyThrows\n     private void decrementCouponTemplateStockAndSaveUserCouponList(CouponTemplateDistributionEvent event) {\n         \/\/ \u5982\u679c\u7b49\u4e8e 0 \u610f\u5473\u7740\u5df2\u7ecf\u6ca1\u6709\u4e86\u5e93\u5b58\uff0c\u76f4\u63a5\u8fd4\u56de\u5373\u53ef\n         Integer couponTemplateStock = decrementCouponTemplateStock(event, event.getBatchUserSetSize());\n         if (couponTemplateStock &lt;= 0) {\n             return;\n         }\n \u200b\n         \/\/ \u83b7\u53d6 Redis \u4e2d\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\n         String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n         List&lt;String&gt; batchUserMaps = stringRedisTemplate.opsForSet().pop(batchUserSetKey, couponTemplateStock);\n \u200b\n         \/\/ \u56e0\u4e3a batchUserIds \u6570\u636e\u8f83\u591a\uff0cArrayList \u4f1a\u8fdb\u884c\u6570\u6b21\u6269\u5bb9\uff0c\u4e3a\u4e86\u907f\u514d\u989d\u5916\u6027\u80fd\u6d88\u8017\uff0c\u76f4\u63a5\u521d\u59cb\u5316 batchUserIds \u5927\u5c0f\u7684\u6570\u7ec4\n         List&lt;UserCouponDO&gt; userCouponDOList = new ArrayList&lt;&gt;(batchUserMaps.size());\n         Date now = new Date();\n \u200b\n         \/\/ \u6784\u5efa userCouponDOList \u7528\u6237\u4f18\u60e0\u5238\u6279\u91cf\u6570\u7ec4\n         for (String each : batchUserMaps) {\n             JSONObject userIdAndRowNumJsonObject = JSON.parseObject(each);\n             DateTime validEndTime = DateUtil.offsetHour(now, JSON.parseObject(event.getCouponTemplateConsumeRule()).getInteger(\"validityPeriod\"));\n             UserCouponDO userCouponDO = UserCouponDO.builder()\n                     .id(IdUtil.getSnowflakeNextId())\n                     .couponTemplateId(event.getCouponTemplateId())\n                     .rowNum(userIdAndRowNumJsonObject.getInteger(\"rowNum\"))\n                     .userId(userIdAndRowNumJsonObject.getLong(\"userId\"))\n                     .receiveTime(now)\n                     .receiveCount(1) \/\/ \u4ee3\u8868\u7b2c\u4e00\u6b21\u9886\u53d6\u8be5\u4f18\u60e0\u5238\n                     .validStartTime(now)\n                     .validEndTime(validEndTime)\n                     .source(CouponSourceEnum.PLATFORM.getType())\n                     .status(CouponStatusEnum.EFFECTIVE.getType())\n                     .createTime(new Date())\n                     .updateTime(new Date())\n                     .delFlag(0)\n                     .build();\n             userCouponDOList.add(userCouponDO);\n         }\n \u200b\n         \/\/ \u5e73\u53f0\u4f18\u60e0\u5238\u6bcf\u4e2a\u7528\u6237\u9650\u9886\u4e00\u6b21\u3002\u6279\u91cf\u65b0\u589e\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\uff0c\u5e95\u5c42\u901a\u8fc7\u9012\u5f52\u65b9\u5f0f\u76f4\u5230\u5168\u90e8\u65b0\u589e\u6210\u529f\n         batchSaveUserCouponList(event.getCouponTemplateId(), event.getCouponTaskBatchId(), userCouponDOList);\n \u200b\n         \/\/ \u5c06\u8fd9\u4e9b\u4f18\u60e0\u5238\u6dfb\u52a0\u5230\u7528\u6237\u7684\u9886\u5238\u8bb0\u5f55\u4e2d\n         List&lt;String&gt; userIdList = userCouponDOList.stream()\n                 .map(UserCouponDO::getUserId)\n                 .map(String::valueOf)\n                 .toList();\n         String userIdsJson = new ObjectMapper().writeValueAsString(userIdList);\n \u200b\n         List&lt;String&gt; couponIdList = userCouponDOList.stream()\n                 .map(each -&gt; StrUtil.builder()\n                         .append(event.getCouponTemplateId())\n                         .append(\"_\")\n                         .append(each.getId())\n                         .toString())\n                 .map(String::valueOf)\n                 .toList();\n         String couponIdsJson = new ObjectMapper().writeValueAsString(couponIdList);\n \u200b\n         \/\/ \u8c03\u7528 Lua \u811a\u672c\u65f6\uff0c\u4f20\u9012\u53c2\u6570\n         List&lt;String&gt; keys = Arrays.asList(\n                 \/\/ \u4e3a\u4ec0\u4e48\u8981\u8fdb\u884c\u66ff\u6362 %s \u4e3a\u7a7a\u767d\u5b57\u7b26\u4e32\uff1f\u56e0\u4e3a\u540e\u7eed\u4ee3\u7801\u9700\u8981\u4f7f\u7528 %s \u8fdb\u884c\u52a8\u6001\u503c\u66ff\u6362\uff0c\u4f46\u662f\u5f53\u524d LUA \u811a\u672c\u4e2d\u4e0d\u9700\u8981\uff0c\u6240\u4ee5\u4e3a\u4e86\u517c\u5bb9\u540e\u7eed\u4e0d\u6539\u52a8\u7279\u6b64\u66ff\u6362\n                 StrUtil.replace(EngineRedisConstant.USER_COUPON_TEMPLATE_LIST_KEY, \"%s\", \"\"),\n                 USER_COUPON_TEMPLATE_LIMIT_KEY,\n                 String.valueOf(event.getCouponTemplateId())\n         );\n         List&lt;String&gt; args = Arrays.asList(\n                 userIdsJson,\n                 couponIdsJson,\n                 String.valueOf(new Date().getTime()),\n                 String.valueOf(event.getValidEndTime().getTime())\n         );\n \u200b\n         \/\/ \u83b7\u53d6 LUA \u811a\u672c\uff0c\u5e76\u4fdd\u5b58\u5230 Hutool \u7684\u5355\u4f8b\u7ba1\u7406\u5bb9\u5668\uff0c\u4e0b\u6b21\u76f4\u63a5\u83b7\u53d6\u4e0d\u9700\u8981\u52a0\u8f7d\n         DefaultRedisScript&lt;Void&gt; buildLuaScript = Singleton.get(BATCH_SAVE_USER_COUPON_LUA_PATH, () -&gt; {\n             DefaultRedisScript&lt;Void&gt; redisScript = new DefaultRedisScript&lt;&gt;();\n             redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource(BATCH_SAVE_USER_COUPON_LUA_PATH)));\n             redisScript.setResultType(Void.class);\n             return redisScript;\n         });\n         stringRedisTemplate.execute(buildLuaScript, keys, args.toArray());\n       \n         \/\/ \u589e\u52a0\u5e93\u5b58\u56de\u6eda\u65b9\u6848\uff0c\u5982\u679c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u4f18\u60e0\u5238\u88ab\u6821\u9a8c\uff0c\u9700\u8981\u5c06 Redis \u9884\u6263\u51cf\u5e93\u5b58\u56de\u6eda\n         int originalUserCouponSize = batchUserMaps.size();\n         \/\/ \u5982\u679c\u7528\u6237\u5df2\u9886\u53d6\u88ab\u6821\u9a8c\u4f1a\u4ece\u96c6\u5408\u4e2d\u5220\u9664\n         int availableUserCouponSize = userCouponDOList.size();\n         int rollbackStock = originalUserCouponSize - availableUserCouponSize;\n         if (rollbackStock &gt; 0) {\n             \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u7f13\u5b58\u5e93\u5b58\u6570\u91cf\n             stringRedisTemplate.opsForHash().increment(\n                     String.format(EngineRedisConstant.COUPON_TEMPLATE_KEY, event.getCouponTemplateId()),\n                     \"stock\",\n                     rollbackStock\n             );\n \u200b\n             \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u5e93\u5e93\u5b58\u6570\u91cf\n             couponTemplateMapper.incrementCouponTemplateStock(event.getShopNumber(), event.getCouponTemplateId(), rollbackStock);\n         }\n     }\n \u200b\n     private Integer decrementCouponTemplateStock(CouponTemplateDistributionEvent event, Integer decrementStockSize) {\n         Long couponTemplateId = event.getCouponTemplateId();\n         int decremented = couponTemplateMapper.decrementCouponTemplateStock(event.getShopNumber(), couponTemplateId, decrementStockSize);\n \u200b\n         \/\/ \u5982\u679c\u4fee\u6539\u8bb0\u5f55\u5931\u8d25\uff0c\u610f\u5473\u7740\u4f18\u60e0\u5238\u5e93\u5b58\u5df2\u4e0d\u8db3\uff0c\u9700\u8981\u91cd\u8bd5\u83b7\u53d6\u5230\u53ef\u81ea\u51cf\u7684\u5e93\u5b58\u6570\u503c\n         if (!SqlHelper.retBool(decremented)) {\n             LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class)\n                     .eq(CouponTemplateDO::getShopNumber, event.getShopNumber())\n                     .eq(CouponTemplateDO::getId, couponTemplateId);\n             CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper);\n             return decrementCouponTemplateStock(event, couponTemplateDO.getStock());\n         }\n \u200b\n         return decrementStockSize;\n     }\n \u200b\n     private void batchSaveUserCouponList(Long couponTemplateId, Long couponTaskBatchId, List&lt;UserCouponDO&gt; userCouponDOList) {\n         \/\/ MyBatis-Plus \u6279\u91cf\u6267\u884c\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\n         try {\n             userCouponMapper.insert(userCouponDOList, userCouponDOList.size());\n         } catch (Exception ex) {\n             Throwable cause = ex.getCause();\n             if (cause instanceof BatchExecutorException) {\n                 \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n                 List&lt;CouponTaskFailDO&gt; couponTaskFailDOList = new ArrayList&lt;&gt;();\n                 List&lt;UserCouponDO&gt; toRemove = new ArrayList&lt;&gt;();\n \u200b\n                 \/\/ \u8c03\u7528\u6279\u91cf\u65b0\u589e\u5931\u8d25\u540e\uff0c\u4e3a\u4e86\u907f\u514d\u5927\u91cf\u91cd\u590d\u5931\u8d25\uff0c\u6211\u4eec\u901a\u8fc7\u65b0\u589e\u5355\u6761\u8bb0\u5f55\u65b9\u5f0f\u6267\u884c\n                 userCouponDOList.forEach(each -&gt; {\n                     try {\n                         userCouponMapper.insert(each);\n                     } catch (Exception ignored) {\n                         \/\/ ...... \u65b0\u589e\u5230 t_coupon_task_fail \u8868\n                 });\n \u200b\n                 \/\/ \u5220\u9664\u5df2\u7ecf\u91cd\u590d\u7684\u5185\u5bb9\n                 userCouponDOList.removeAll(toRemove);\n                 return;\n             }\n \u200b\n             throw ex;\n         }\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u68b3\u7406\u8fd9\u6bb5\u4ee3\u7801\u4e1a\u52a1\u903b\u8f91\u5982\u4e0b\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>1. \u9996\u5148\u8c03\u7528 <code>decrementCouponTemplateStock<\/code> \u8fdb\u884c\u6263\u51cf MySQL \u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\uff0c\u5982\u679c\u4e00\u6b21\u6027\u6263\u51cf\u6210\u529f\uff0c\u90a3\u4e48\u8fdb\u884c\u4e0b\u8ff0\u903b\u8f91\uff1b\u5982\u679c\u6263\u51cf\u5931\u8d25\uff0c\u4f1a\u6709\u4e00\u4e2a\u9012\u5f52\u6263\u51cf\uff0c\u76f4\u5230\u6263\u51cf\u6307\u5b9a\u7684\u5e93\u5b58\u4e3a\u6b62\uff1b<\/li>\n\n\n\n<li>2. \u83b7\u53d6 Redis \u4e2d\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238 Set \u5217\u8868\uff0c\u8f6c\u6362\u6210 userCouponDOList \u5f85\u4fdd\u5b58\u5230\u6570\u636e\u5e93\u96c6\u5408\uff1b<\/li>\n\n\n\n<li>3. \u8c03\u7528 <code>batchSaveUserCouponList<\/code> \u6279\u91cf\u4fdd\u5b58\u65b9\u6cd5\uff0c\u5982\u679c\u6ca1\u6709\u552f\u4e00\u7d22\u5f15\u51b2\u7a81\uff0c\u5219\u65b9\u6cd5\u7ed3\u675f\uff0c\u5426\u5219\u6309\u7167\u5355\u6761\u6267\u884c\u65b0\u589e\u8bed\u53e5\uff1b<\/li>\n\n\n\n<li>4. \u6279\u91cf\u4fdd\u5b58\u5230\u6570\u636e\u5e93\u540e\uff0c\u6211\u4eec\u5c06\u4fdd\u5b58\u6210\u529f\u7684\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\u65b0\u589e\u5230\u7528\u6237 Redis \u9886\u5238\u8bb0\u5f55\u4e2d\uff1b<\/li>\n\n\n\n<li>5. \u5982\u679c\u8bf4\u7528\u6237\u5df2\u7ecf\u9886\u8fc7\u5238\uff0c\u9700\u8981\u5c06\u5df2\u7ecf\u9884\u6263\u51cf\u8fc7\u7684\u5e93\u5b58\u518d\u8c03\u7528 <code>incrementCouponTemplateStock<\/code> \u65b9\u6cd5\u8fdb\u884c\u56de\u6eda\u3002<\/li>\n<\/ul>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"1011\" height=\"662\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388051-image.png\" alt=\"\" class=\"wp-image-1383\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388051-image.png 1011w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388051-image-300x196.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388051-image-768x503.png 768w\" sizes=\"auto, (max-width: 1011px) 100vw, 1011px\" \/><\/figure>\n<\/blockquote>\n\n\n\n<p>\u6240\u4ee5\u7684\u8bdd\uff0c\u6211\u4eec\u5c31\u4e0d\u5148\u6253\u65ad\u70b9\u4e86\u3002\u56e0\u4e3a\u521a\u624d\u6211\u6253\u65ad\u70b9\u7684\u65f6\u5019\uff0c\u5176\u5b9e\u53d1\u73b0\u4e86\u4e00\u4e9b\u95ee\u9898\u3002<\/p>\n\n\n\n<p>\u9996\u5148\uff0c\u6211\u4eec\u521a\u624d\u7684\u4f18\u60e0\u5238 Excel \u91cc\u9762\u5176\u5b9e\u662f 5000 \u6761\u3002\u7b2c 5000 \u6b21\u7684\u8bdd\uff0c\u5b83\u4f1a\u901a\u8fc7\u4e0a\u9762\u8fd9\u4e2a\u903b\u8f91\u53bb\u5f80\u8fd9\u91cc\u53d1\u4e2a\u6d88\u606f\u3002\u7136\u540e\u53d1\u6d88\u606f\u7684\u65f6\u5019\uff0c\u5b83\u5176\u5b9e\u5df2\u7ecf\u53d1\u51fa\u53bb\u4e86\uff0c\u5bf9\u5427\uff1f\u7136\u540e\u5728\u8fd9\u4e2a Excel \u91cc\u9762\uff0c\u5b83\u5c31\u4f1a\u5f00\u59cb\u6267\u884c\u8fd9\u4e00\u6b65\u903b\u8f91\u3002<\/p>\n\n\n\n<figure class=\"wp-block-image size-full\"><img loading=\"lazy\" decoding=\"async\" width=\"970\" height=\"625\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388092-image.png\" alt=\"\" class=\"wp-image-1385\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388092-image.png 970w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388092-image-300x193.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759388092-image-768x495.png 768w\" sizes=\"auto, (max-width: 970px) 100vw, 970px\" \/><\/figure>\n\n\n\n<p>\u7136\u540e\u8fd9\u4e2a\u662f\u5728\u5b83 Excel \u6267\u884c\u89e3\u6790\u5b8c\u4e86\uff0c\u5bf9\u5427\uff1f\u5b83\u8fd8\u4f1a\u518d\u53d1\u4e00\u4e2a\u201c\u7ed3\u675f\u201d\u7684\u6807\u8bc6\uff0c\u610f\u5473\u7740\u201c\u6211 Excel \u5df2\u7ecf\u89e3\u6790\u6210\u529f\u4e86\u201d\uff0c\u7136\u540e\u518d\u53bb\u544a\u8bc9\u6211\u6211\u7684\u6d88\u8d39\u8005\uff1a\u201c\u4f60\u6709\u4e00\u4e9b\u6d88\u606f\uff0c\u53ef\u80fd\u6ca1\u6709\u6ee1 5000\uff0c\u5176\u5b9e\u4ed6\u5df2\u7ecf\u5728\u5bf9\u5e94\u7684\u7f13\u5b58\u91cc\u9762\u4e86\u3002\u53d1\u8fd9\u4e2a\u6d88\u606f\u5c31\u662f\u8981\u628a\u90a3\u4e9b\u7ed9\u6e05\u7a7a\u6389\u3002\u201d<\/p>\n\n\n\n<p>\u8fd9\u5c31\u4f1a\u5bfc\u81f4\u4e00\u4e2a\u95ee\u9898\uff1a\u6211\u5728\u8fd9\u8fb9\u7b2c\u4e00\u4e2a\u6d88\u606f\u6765\u4e86\u4e4b\u540e\uff0c\u6211\u6253\u4e2a\u65ad\u70b9\uff0c\u5176\u5b9e\u7b2c\u4e8c\u4e2a\u6d88\u606f\u4ed6\u4f1a\u628a\u6211\u7684\u903b\u8f91\u7ed9\u6267\u884c\u6389\u3002<\/p>\n\n\n\n<p>\u6211\u4e3e\u4e2a\u7b80\u5355\u7684\u4f8b\u5b50\uff1a\u90a3\u5c31\u662f\u6211\u4eec\u5728\u8fd9\u91cc\uff0c\u5b83\u6709\u4e00\u4e2a\u5224\u65ad<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u5f53\u4fdd\u5b58\u7528\u6237\u4f18\u60e0\u5238\u96c6\u5408\u8fbe\u5230\u6279\u91cf\u4fdd\u5b58\u6570\u91cf\nCouponTemplateDistributionEvent event = messageWrapper.getMessage();\nif (!event.getDistributionEndFlag() &amp;&amp; event.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0) {\n    decrementCouponTemplateStockAndSaveUserCouponList(event);\n    return;\n}<\/code><\/pre>\n\n\n\n<p>\u5728CouponTemplateDistributionEvent event = messageWrapper.getMessage();\u6253\u65ad\u70b9 \uff0c\u5982\u679c\u8bf4\u5b83\u4e0d\u662f\u89e3\u6790\u5b8c\u6210\uff0c\u5e76\u4e14\u5b83\u7684\u6279\u91cf\u6279\u6b21\u5df2\u7ecf\u8fbe\u5230\u4e86 5000\uff0c\u6211\u4eec\u4f1a\u6267\u884c\u8fd9\u4e2a\u903b\u8f91\u3002OK\uff0c\u6211\u5728\u8fd9\u91cc\u6253\u65ad\u70b9\u3002\u7136\u540e\u7b2c\u4e8c\u4e2a\u6d88\u606f\u4ed6\u53d1\u8fc7\u6765\u7684\u65f6\u5019\uff0c\u5176\u5b9e\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\uff0c\u5b8c\u6210\u6807\u8bc6\u5df2\u7ecf\u5230\u4e86\u8fd9\u91cc\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>@Override\npublic void doAfterAllAnalysed(AnalysisContext context) {\n    \/\/ \u53d1\u9001 Excel \u89e3\u6790\u5b8c\u6210\u6807\u8bc6\uff0c\u5373\u4f7f\u4e0d\u6ee1\u8db3\u6279\u91cf\u4fdd\u5b58\u7684\u6570\u91cf\u4e5f\u5f97\u4fdd\u5b58\u5230\u6570\u636e\u5e93\n    CouponTemplateDistributionEvent couponTemplateExecuteEvent = CouponTemplateDistributionEvent.builder()\n            .distributionEndFlag(Boolean.TRUE) \/\/ \u8bbe\u7f6e\u89e3\u6790\u5b8c\u6210\u6807\u8bc6\n            .shopNumber(couponTemplateD00.getShopNumber())\n            .couponTemplateId(couponTemplateD00.getId())\n            .validEndTime(couponTemplateD00.getValidEndTime())\n            .couponTemplateConsumeRule(couponTemplateD00.getConsumeRule())\n            .couponTaskBatchId(couponTaskD00.getBatchId())\n            .couponTaskId(couponTaskD00.getId())\n            .build();\n    couponExecuteDistributionProducer.sendMessage(couponTemplateExecuteEvent);\n}<\/code><\/pre>\n\n\n\n<p>\u4ed6\u53d1\u73b0\u5df2\u7ecf\u6ca1\u6709 Excel \u8bb0\u5f55\u7684\u65f6\u5019\uff0c\u4ed6\u4f1a\u53bb\u628a\u6211\u6240\u6709\u7684\u7f13\u5b58\u91cc\u9762\u7684\u8bb0\u5f55\u7ed9\u4ed6\u5f39\u51fa\u6765\u2014\u2014\u770b\u5230\u6ca1\u6709\uff1f\u4ed6\u4f1a\u53bb\u628a set \u5f39\u51fa\u6765\uff0c\u5f39\u7684\u6570\u636e\u91cf\u662f\u6700\u5927\u7684\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6\u4e3a TRUE\uff0c\u4ee3\u8868\u5df2\u7ecf\u6ca1\u6709 Excel \u8bb0\u5f55\u4e86\nif (event.getDistributionEndFlag()) {\n    String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n    Long batchUserIdsSize = stringRedisTemplate.opsForSet().size(batchUserSetKey);\n    event.setBatchUserSetSize(batchUserIdsSize.intValue());\n\n    decrementCouponTemplateStockAndSaveUserCouponList(event);\n    List&lt;String>\n    \/\/ \u6b64\u65f6\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\u5982\u679c\u8fd8\u6709\u503c\uff0c\u5c31\u610f\u5473\u7740\u53ef\u80fd\u5e93\u5b58\u4e0d\u8db3\u5f15\u8d77\n    if (CollUtil.isNotEmpty(batchUserMaps)) {<\/code><\/pre>\n\n\n\n<p>\u5927\u5bb6\u80fd\u7406\u89e3\u6211\u8bf4\u7684\u5417\uff1f\u6bd4\u5982\u8bf4\uff0c\u6211\u6b63\u5e38\u5728\u8fd9\u4e2a\u8282\u70b9\uff0c\u6211\u7b2c\u4e00\u4e2a\u6d88\u606f\u6765\u4e86\uff0c\u6211\u8981\u6263\u51cf 5000\uff0c\u5bf9\u5427\uff1f\u6211\u6570\u636e\u5e93\u91cc\u9762\u662f 5001\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u6211\u5728\u8fd9\u8fb9\u6253\u7740\u65ad\u70b9\uff0c\u7136\u540e\u4ed6\u7ed9\u6211\u6267\u884c\u5b8c\u4e86\uff0c\u76f8\u5f53\u4e8e\u6211\u6570\u636e\u5e93\u91cc\u9762\u53ea\u6709\u4e00\u4e2a\u5e93\u5b58\u4e86\u2014\u2014\u8fd9\u5c31\u6709\u95ee\u9898\u4e86\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u7136\u540e\u63a5\u4e0b\u6765\u7684\u8bdd\uff0c\u6211\u901a\u8fc7\u65ad\u70b9\u7684\u5f62\u5f0f\u5c31\u662f\u4e00\u6539\u3002\u6211\u8ddf\u5927\u5bb6\u5148\u68b3\u7406\u4e00\u904d\u903b\u8f91\uff0c\u5927\u5bb6\u81ea\u5df1\u53ef\u4ee5\u5728\u672c\u5730\u7535\u8111\u4e0a\u53bb\u6253\u65ad\u70b9\u53bb\u8fdb\u884c\u4e00\u4e2a\u8c03\u8bd5\u3002\u7b2c\u4e00\u5757\uff0c\u9996\u5148\u6211\u4eec\u8981\u53bb\u6263\u51cf\u6211\u4eec\u7684\u4f18\u60e0\u5238\u6a21\u677f\u7684\u5e93\u5b58\uff1b<br>\u5176\u6b21\uff0c\u6211\u4eec\u8981\u53bb\u4fdd\u5b58\u6211\u4eec\u7684\u7528\u6237\u9886\u5238\u8bb0\u5f55\u3002<\/p>\n\n\n\n<p>OK\uff0c\u7b2c\u4e00\u70b9\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u53bb\u6570\u636e\u5e93\u91cc\u9762\u53bb\u8fdb\u884c\u4e00\u4e2a\u81ea\u51cf\u3002\u5927\u5bb6\u4e4b\u524d\u4e5f\u90fd\u770b\u8fc7\uff0c\u90a3\u5c31\u662f\u4e00\u4e2a\u901a\u8fc7\u60b2\u89c2\u9501\u7684\u5f62\u5f0f\u53bb\u7ed9\u4ed6\u4e00\u4e2a\u5e76\u53d1\u4fdd\u969c\u7684\u6263\u51cf\u3002<code>&lt;!-- \u901a\u8fc7 MySQL \u60b2\u89c2\u884c\u8bb0\u5f55\u9501\u786e\u4fdd\u5e93\u5b58\u4e0d\u4f1a\u88ab\u591a\u6263\uff0c\u5e76\u91c7\u7528\u4e0a\u5c42\u4e1a\u52a1\u81ea\u65cb\u91cd\u8bd5\u76f4\u81f3\u6210\u529f\u5c06\u4f18\u60e0\u5238\u5e93\u5b58\u6263\u51cf\u81f3\u96f6 --> &lt;update id=\"decrementCouponTemplateStock\"> UPDATE t_coupon_template SET stock = stock - #{decrementStock} WHERE shop_number = #{shopNumber} AND id = #{couponTemplateId} AND stock >= #{decrementStock} &lt;\/update> &lt;!-- \u56de\u6eda MySQL \u6570\u636e\u5e93\u4f18\u60e0\u5238\u5e93\u5b58\u4f59\u91cf\uff0c\u901a\u8fc7 MySQL \u60b2\u89c2\u884c\u8bb0\u5f55\u9501\u4fdd\u969c\u5e76\u53d1\u5b89\u5168 --> &lt;update id=\"incrementCouponTemplateStock\"> UPDATE t_coupon_template SET stock = stock + #{incrementStock} WHERE shop_number = #{shopNumber} AND id = #{couponTemplateId} &lt;\/update><\/code><\/p>\n\n\n\n<p>\u7136\u540e\u5728\u8fd9\u91cc\uff0c\u5982\u679c\u8bf4\u2014\u2014\u56e0\u4e3a\u6211\u4eec\u53bb\u64cd\u4f5c MySQL \u7684\u65f6\u5019\uff0c\u5982\u679c\u4f60\u52a0\u4e2a\u8fd4\u56de\u503c\u7684\u8bdd\uff0c\u5b83\u662f\u4e2a int \u7c7b\u578b\u3002\u5982\u679c\u8bf4\u5bf9\u8fd9\u6761\u8bb0\u5f55\u6709\u53d8\u66f4\uff0c\u5b83\u5c31\u4f1a\u8fd4\u56de 1\uff1b\u5982\u679c\u8bf4\u6ca1\u6709\u53d8\u66f4\uff0c\u5b83\u5c31\u4f1a\u8fd4\u56de 0\u3002<\/p>\n\n\n\n<p>\u8fd9\u91cc\u5982\u679c\u8fd4\u56de 0 \u7684\u8bdd\uff0c\u5c31\u4e00\u5b9a\u662f\u610f\u5473\u7740\u6211\u4eec\u7684\u5e93\u5b58\u4e0d\u8db3\u4e86\u3002\u56e0\u4e3a\u6211\u4eec\u5728\u8fd9\u91cc\u9762\u6709\u4e00\u4e2a\u5224\u65ad\uff1a\u6211\u4eec\u8981\u6263\u51cf\u5e93\u5b58\uff0c\u4e00\u5b9a\u8981\u5927\u4e8e\u7b49\u4e8e\u5b83\u8981\u6263\u51cf\u7684\u6570\u91cf\u3002\u6240\u4ee5\u8bf4\uff0c\u8fd9\u91cc\u5224\u65ad\u4e0d\u8db3\u7684\u8bdd\uff0c\u6211\u4eec\u5c31\u8981\u5f00\u59cb\u83b7\u53d6\u5230\u6700\u65b0\u7684\u4e00\u4e2a\u5e93\u5b58\uff0c\u7136\u540e\u518d\u7ed9\u4ed6\u6267\u884c\u4e00\u6b21\u6211\u4eec\u7684\u9012\u5f52\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u76f4\u5230\u80fd\u6263\u51cf\u51fa\u6765\u5b83\u5bf9\u5e94\u7684\u4e00\u4e2a\u80fd\u591f\u88ab\u6263\u51cf\u7684\u5e93\u5b58\u6570\u91cf\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u8fd4\u56de\u6b63\u5e38\u7684\u8bdd\uff0c\u6211\u4eec\u8fd9\u91cc\u9762\u80af\u5b9a\u662fdecrementStockSize\u548c\u8fd4\u56de\u503c\u4e00\u5b9a\u662f\u76f8\u7b49\u7684\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u5f02\u5e38\u7684\u60c5\u51b5\u4e0b\u624d\u4f1a\u4e0d\u7b49\u3002<\/p>\n\n\n\n<p>OK\uff0c\u5047\u8bbe\u6211\u4eec\u662f\u76f8\u7b49\u7684\uff0c\u7136\u540e\u5230\u4e86\u4e0a\u9762\u4e4b\u540e\uff0c\u5b83\u5e94\u8be5\u662f 5000\uff0c\u5bf9\u5427\uff1f\u5728\u8fd9\u91cc\uff0cbatchUserSetKey\u662f\u6211\u4eec\u5728 Redis \u5e95\u5c42\u91cc\u9762\u4fdd\u5b58\u7684\u7f13\u51b2\u961f\u5217\u3002\u7136\u540e\u6211\u4eec\u53bb\u62ff\u5230\u5df2\u7ecf\u6263\u51cf\u6210\u529f\u7684\u5e93\u5b58\u2014\u2014\u76f8\u5f53\u4e8e\u6211\u4eec\u5df2\u7ecf\u6709\u4e86\u4e00\u4e2a\u5165\u573a\u5238\u4e86\u3002<\/p>\n\n\n\n<p>\u63a5\u4e0b\u6765\u7684\u8bdd\uff0c\u6211\u4eec\u5df2\u7ecf\u628a\u4ece\u7f13\u51b2\u961f\u5217\u91cc\u9762\u628a\u6570\u636e\u7ed9\u5f39\u51fa\u6765\u4e86\uff0c\u7136\u540e\u6211\u4eec\u8981\u53bb\u628a\u8fd9\u4e9b\u6570\u636e\u7ed9\u5b83\u65b0\u589e\u5230\u6211\u4eec\u7684\u6570\u636e\u5e93\u91cc\u9762\u3002<\/p>\n\n\n\n<p>\u5728\u8fd9\u91cc\uff0c\u6211\u4eec\u662f\u7ed9\u4ed6\u8fdb\u884c\u4e86\u4e00\u4e9b\u7ec4\u88c5\uff0c\u903b\u8f91\u6bd4\u8f83\u7b80\u5355\u4e00\u70b9\uff0c\u6211\u4eec\u5c31\u5148\u8df3\u8fc7\u3002\u5c31\u90a3\u9640for\u91cc\u9762\u7684\u5185\u5bb9<\/p>\n\n\n\n<p>\u7136\u540e\u53bb\u5728\u8fd9\u91cc\u9762\u6709\u4e00\u4e2a\u7279\u6b8a\u8bf4\u660e<br>batchSaveUserouponList\uff08event.getCouponTemplateId(),event.getCouponTaskBatchId(),userCouponDoList);<br>\uff1a\u6211\u4eec\u53bb\u9650\u5236\u4e86\u6bcf\u4e2a\u7528\u6237\u53ea\u80fd\u9650\u9886\u4e00\u6b21\u3002\u7136\u540e\u5728\u8fd9\u91cc\uff0c\u8fd9\u662f\u4e00\u4e2a\u6279\u91cf\u65b0\u589e\u7684\u65b9\u6cd5\u2014\u2014\u5c31\u662f MyBatis-Plus \u5b83\u7684\u4e00\u4e2a\u6269\u5c55\u65b9\u6cd5\u3002<\/p>\n\n\n\n<p>\u7b2c\u4e00\u4e2auserCouponMapper.insert(userCouponDoList,userLouponDoList.size();\u662f\u6211\u4eec\u7684\u4e00\u4e2a\u9700\u8981\u6279\u91cf\u65b0\u589e\u7684\u6570\u636e\u96c6\u5408\uff0c\u7b2c\u4e8c\u5757\u662f\u5b83\u7684\u4e00\u4e2a size\u2014\u2014\u76f8\u5f53\u4e8e\u6211\u4eec\u5355\u6b21\u7684\u6279\u91cf\u7684\u6570\u503c\u3002\u6211\u4eec\u76f8\u5f53\u4e8e\u6211\u4eec\u662f 5000\uff0c\u5c31\u76f4\u63a5\u8ba9\u4ed6\u6279\u91cf\u65b0\u589e\u5c31\u597d\u4e86\u3002\u4ed6\u9ed8\u8ba4\u7684\u8bdd\uff0c\u6211\u5982\u679c\u8bb0\u5f97\u6ca1\u9519\uff0c\u5e94\u8be5\u662f\u5355\u6b21 1000\u3002<\/p>\n\n\n\n<p>\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\u3002\u5bf9\uff0c\u9ed8\u8ba4\u7684\u8bdd 1000\u3002\u5982\u679c\u8bf4\u6211\u4eec\u4e0d\u8bbe\u7f6e\u7684\u8bdd\uff0c\u76f8\u5f53\u4e8e\u6211\u4eec\u4e00\u6b21 5000 \u7684\u8bb0\u5f55\uff0c\u4ed6\u4f1a\u5206 5 \u6b21\u6267\u884c\uff0c\u6027\u80fd\u4e0d\u5c31\u6162\u4e86 5 \u500d\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p><\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>Q\uff1aevent.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0 \u8fd9\u6bb5\u4ee3\u7801\u7684\u610f\u4e49\u662f\u4ec0\u4e48\uff1f<\/p>\n\n\n\n<p>A\uff1a\u4e4b\u524d\u5176\u5b9e\u6ca1\u6709\u8fd9\u884c\u4ee3\u7801\u7684\uff0c\u5f97\u76ca\u4e8e\u661f\u7403\u540c\u5b66\u7684\u63d0\u9192\uff0c\u6211\u4eec\u5c06\u8fd9\u4e2a\u903b\u8f91\u8fdb\u884c\u4e86\u6dfb\u52a0\u3002\u5982\u679c BatchSize \u7b49\u4e8e 50001\uff0c\u90a3\u4e48\u53d6\u6a21\u540e\u503c\u4e0d\u7b49\u4e8e 0\uff0c\u5c31\u4f1a\u8df3\u8fc7\u8be5\u6d41\u7a0b\u3002\u76f4\u5230\u4e0b\u6b21 BatchSize \u7b49\u4e8e 5000 \u65f6\u6267\u884c\u3002<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"766\" width=\"1614\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/60d364a159317344a842065bd6ef3f59.png\" alt=\"image-20240908000620084.png\"><\/p>\n\n\n\n<p>\u53e6\u5916\uff0c\u6211\u89c9\u5f97\u5927\u5bb6\u53ef\u80fd\u5bf9\u6279\u91cf\u4fdd\u5b58\u8fd9\u91cc\u4f1a\u6709\u70b9\u7591\u95ee\uff0c\u4e4b\u524d\u6211\u4eec\u662f\u7528\u9012\u5f52\u7684\u5f62\u5f0f\u5199\u7684\uff0c\u4e3a\u4ec0\u4e48\u6539\u6210\u6279\u91cf\u65b0\u589e\u5931\u8d25\u540e\u9000\u5316\u6210\u5355\u6761\u65b0\u589e\uff1f<\/p>\n\n\n\n<p>\u5982\u679c\u6211\u4eec\u4f7f\u7528\u9012\u5f52\u7684\u8bdd\uff0c\u4f1a\u6709\u4e00\u4e2a\u5f88\u5c34\u5c2c\u5f88\u5c34\u5c2c\u7684\u95ee\u9898\uff0c\u4e0d\u77e5\u9053\u662f\u4e8b\u52a1\u5f15\u8d77\u7684\uff0c\u8fd8\u662f\u6279\u91cf\u65b0\u589e\u5f15\u8d77\u7684\u3002\u8fd9\u4e2a\u662f\u4e4b\u524d\u7684\u4ee3\u7801\uff1a<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> private void batchSaveUserCouponList(Long couponTemplateId, List&lt;UserCouponDO&gt; userCouponDOList) {\n     \/\/ MyBatis-Plus \u6279\u91cf\u6267\u884c\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\n     try {\n         userCouponMapper.insert(userCouponDOList, userCouponDOList.size());\n     } catch (Exception ex) {\n         Throwable cause = ex.getCause();\n         if (cause instanceof BatchExecutorException) {\n             \/\/ \u67e5\u8be2\u5df2\u7ecf\u5b58\u5728\u7684\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\n             List&lt;Long&gt; userIds = userCouponDOList.stream().map(UserCouponDO::getUserId).toList();\n             List&lt;UserCouponDO&gt; existingUserCoupons = getExistingUserCoupons(couponTemplateId, userIds);\n             \/\/ \u904d\u5386\u5df2\u7ecf\u5b58\u5728\u7684\u96c6\u5408\uff0c\u83b7\u53d6 userId\uff0c\u5e76\u4ece\u9700\u8981\u65b0\u589e\u7684\u96c6\u5408\u4e2d\u79fb\u9664\u5339\u914d\u7684\u5143\u7d20\n             for (UserCouponDO each : existingUserCoupons) {\n                 Long userId = each.getUserId();\n \u200b\n                 \/\/ \u4f7f\u7528\u8fed\u4ee3\u5668\u904d\u5386\u9700\u8981\u65b0\u589e\u7684\u96c6\u5408\uff0c\u5b89\u5168\u79fb\u9664\u5143\u7d20\n                 Iterator&lt;UserCouponDO&gt; iterator = userCouponDOList.iterator();\n                 while (iterator.hasNext()) {\n                     UserCouponDO item = iterator.next();\n                     if (item.getUserId().equals(userId)) {\n                         iterator.remove();\n                     }\n                 }\n             }\n \u200b\n             \/\/ \u91c7\u7528\u9012\u5f52\u65b9\u5f0f\u91cd\u8bd5\uff0c\u76f4\u5230\u4e0d\u5b58\u5728\u91cd\u590d\u7684\u8bb0\u5f55\u4e3a\u6b62\n             if (CollUtil.isNotEmpty(userCouponDOList)) {\n                 batchSaveUserCouponList(couponTemplateId, userCouponDOList);\n             }\n         }\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u95ee\u9898\u590d\u73b0\u6b65\u9aa4\uff1a\u6570\u636e\u5e93\u4e2d\u7684\u7528\u6237\u4f18\u60e0\u5238\u8868\u5df2\u6709\u4e00\u6761\u8bb0\u5f55\uff0cID \u4e3a 1\u3002\u5728\u4e00\u4e2a\u4e8b\u52a1\u4e2d\u4f7f\u7528 MyBatisPlus \u7684\u6279\u91cf\u65b0\u589e\u65b9\u6cd5\u63d2\u5165\u4e09\u6761\u8bb0\u5f55\uff0cID \u5206\u522b\u4e3a 1\u30012\u30013\u3002\u8c03\u7528\u6279\u91cf\u65b0\u589e\u65b9\u6cd5\u65f6\uff0c\u51fa\u73b0 ID 1 \u4e3b\u952e\u91cd\u590d\u7684\u9519\u8bef\u3002\u5c06 ID 1 \u4ece\u96c6\u5408\u4e2d\u5220\u9664\u540e\uff0c\u5728\u540c\u4e00\u4e2a\u65b9\u6cd5\u4e2d\u901a\u8fc7\u9012\u5f52\u518d\u6b21\u8c03\u7528\u6279\u91cf\u65b0\u589e\u65b9\u6cd5\uff0c\u8fd9\u6b21\u5374\u62a5\u9519 ID 2 \u4e3b\u952e\u51b2\u7a81\uff0c\u4f46\u5b9e\u9645\u4e0a\u6570\u636e\u5e93\u4e2d\u5e76\u4e0d\u5b58\u5728 ID 2\u3002\u7531\u4e8e ID 2 \u65e0\u6cd5\u67e5\u8be2\u5230\uff0c\u4e5f\u65e0\u6cd5\u4ece\u96c6\u5408\u4e2d\u5220\u9664\uff0c\u5bfc\u81f4\u9012\u5f52\u8c03\u7528\u6301\u7eed\u62a5\u9519\u3002<\/p>\n\n\n\n<p>\u786e\u5b9e\u6ca1\u627e\u5230\u539f\u56e0\u662f\u4ec0\u4e48\uff0c\u6700\u7ec8\u53ea\u80fd\u6539\u6210\u6279\u91cf\u4fdd\u5b58\u5931\u8d25\u540e\u8f6c\u4e3a\u5355\u6761\u8bb0\u5f55\u4fdd\u5b58\u3002\u5982\u679c\u62a5\u9519\u5219\u5ffd\u7565\uff0c\u6700\u7ec8\u6ca1\u6709\u62a5\u9519\u7684\u8bb0\u5f55\u4f1a\u8ddf\u7740\u4e8b\u52a1\u4e00\u8d77\u63d0\u4ea4\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u8fd9\u91cc\u8bf4\u4e0b\u552f\u4e00\u7d22\u5f15\u7684\u529f\u80fd\uff0c\u5373\u4f7f\u4e8b\u52a1\u6ca1\u6709\u63d0\u4ea4\uff0c\u552f\u4e00\u7d22\u5f15\u4e5f\u662f\u751f\u6548\u7684\u3002\u4e3e\u4f8b\uff1a\u4e8b\u52a1\u6267\u884c\u4e2d\uff0c\u6ca1\u63d0\u4ea4\u6570\u636e\uff1b\u7136\u540e\u5728\u6570\u636e\u5e93\u53ef\u89c6\u5316\u5de5\u5177\u91cc\uff0c\u65b0\u589e\u540c\u6837\u7684\u552f\u4e00\u7d22\u5f15\u8bb0\u5f55\uff0c\u4f1a\u62a5\u9519\u3002<\/p>\n<\/blockquote>\n\n\n\n<p>catch (Exception ex) {<br>         Throwable cause = ex.getCause();<br>         if (cause instanceof BatchExecutorException) {..............<\/p>\n\n\n\n<p><strong>\u91cd\u70b9\u5728\u8fd9\u91cc\uff1a\u5982\u679c\u8bf4\u6211\u4eec\u6279\u91cf\u6267\u884c\u7684\u65f6\u5019\uff0c\u5047\u8bbe\u7528\u6237\u5df2\u7ecf\u9886\u4e86\u4e00\u6b21\u4f18\u60e0\u5238\u4e86\uff0c\u8fd9\u91cc\u5c31\u4f1a\u88ab\u6211\u4eec\u7684\u552f\u4e00\u7d22\u5f15\u7ed9\u6821\u9a8c\u4f4f\u3002\u6821\u9a8c\u4f4f\u4e4b\u540e\uff0c\u6211\u4eec\u5c31\u4f1a\u629b\u4e00\u4e2a\u5f02\u5e38\u51fa\u6765\u3002\u6211\u4eec\u53bb\u8bc6\u522b\uff0c\u7136\u540e\u90a3\u5c31\u662f\uff1a\u6211\u628a\u8fd9\u4e2a\u8bb0\u5f55\u4ece\u4e4b\u524d\u7684\u6279\u91cf\u65b0\u589e\u6539\u6210\u5355\u6b21\u65b0\u589e\u3002\u5982\u679c\u5355\u6b21\u65b0\u589e\u8fd8\u6709\u95ee\u9898\uff0cOK\uff0c\u6211\u53bb\u67e5\u4e00\u4e0b\u5b83\u662f\u5426\u5df2\u7ecf\u5b58\u5728\u4e86\u8fd9\u6761\u8bb0\u5f55\u3002\u5982\u679c\u5df2\u7ecf\u5b58\u5728\uff0c\u597d\uff0c\u6211\u628a\u5b83\u7ed9\u6dfb\u52a0\u5230\u6211\u7684\u7528\u6237\u7684\u9886\u5238\u5931\u8d25\u8bb0\u5f55\u91cc\u9762\u53bb\uff0c\u5e76\u4e14\u6211\u628a\u5b83\u4ece\u6211\u7684\u4e3b\u673a\u96c6\u5408\u91cc\u9762\u7ed9\u5b83\u62b9\u6389\u3002<\/strong><\/p>\n\n\n\n<p>\u56e0\u4e3a\u6211\u4e4b\u524d\u662f\u6d4b\u8bd5\u8fc7\u7684\u6548\u679c\u662f\u597d\u7684\uff0c\u7136\u540e\u6211\u7ed9\u5927\u5bb6\u770b\u4e00\u4e0b\uff1a\u201c\u7528\u6237\u5df2\u9886\u53d6\u8be5\u4f18\u60e0\u5238\u201d\u3002\u5176\u5b9e\u590d\u73b0\u7684\u903b\u8f91\u5f88\u7b80\u5355\uff0c\u90a3\u5c31\u662f\u6211\u4eec\u5728\u6570\u636e\u5e93\u91cc\u9762\u968f\u4fbf\u627e\u4e00\u4e2a\u7528\u6237\u3002\u5982\u679c\u8bf4\u4f60\u4e4b\u524d\u6267\u884c\u8fc7\u5206\u53d1\uff0cOK\uff0c\u50cf\u6211\u8fd9\u79cd\uff0c\u627e\u5230\u7528\u6237\u7684 ID \u4e4b\u540e\uff0c\u6211\u4eec\u62ff\u6700\u65b0\u7684\u4e00\u6b21\u5b83\u7684\u6a21\u677f ID \u7ed9\u5b83\u2014\u2014\u5c31\u662f\u5047\u8bbe\u6211\u5728\u91cc\u9762\u65b0\u589e\u4e00\u4e2a\u6a21\u677f ID\uff0c\u7136\u540e\u62ff\u5230\u4e4b\u540e\u7ed9\u5b83\u653e\u5230\u8fd9\u91cc\uff0c\u5176\u5b9e\u5b83\u5df2\u7ecf\u5b58\u5728\u4e86\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u8fd9\u91cc\u9762\u5c31\u4f1a\u629b\u5f02\u5e38\uff0c\u8fd9\u6761\u8bb0\u5f55\u91cc\u9762\u5c31\u4f1a\u6709\u503c\u4e86\u3002\u5927\u5bb6\u53ef\u4ee5\u53bb\u8bd5\u4e00\u4e0b\uff0c\u7136\u540e\u8fd9\u91cc\u9762\u6211\u4eec\u5c31\u4e0d\u518d\u8fc7\u591a\u8d58\u8ff0\u3002\u7136\u540e\u628a\u7528\u6237\u7684\u4f18\u60e0\u5238\u5df2\u7ecf\u52a0\u5230\u6570\u636e\u5e93\u4e4b\u540e\u4e86\uff0c\u6211\u4eec\u9700\u8981\u628a\u8fd9\u4e2a\u6570\u636e\u7ed9\u5b83\u9884\u70ed\u5230\u7f13\u5b58\u91cc\u9762 <\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"543\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759391940-image-1024x543.png\" alt=\"\" class=\"wp-image-1388\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759391940-image-1024x543.png 1024w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759391940-image-300x159.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759391940-image-768x407.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1759391940-image.png 1028w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/figure>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u4e4b\u524d\u53ef\u4ee5\u770b\u5230\uff0c\u6211\u4eec\u662f\u901a\u8fc7 Lua \u811a\u672c\u53bb\u505a\u7684\u3002\u56e0\u4e3a\u4e00\u822c\u6765\u8bb2\uff0c\u9762\u5bf9\u4e00\u4e9b To C \u6216\u8005\u8bf4\u4e00\u4e9b\u9ad8\u5e76\u53d1\u7684\u903b\u8f91\uff0c\u5c3d\u91cf\u8fd8\u662f\u7528 Redis \u811a\u672c\u53bb\u505a\uff0c\u6216\u8005\u8bf4\u7ba1\u9053\u7684\u5f62\u5f0f\uff0c\u8fd9\u6837\u7684\u8bdd\u6027\u80fd\u4f1a\u9ad8\u4e00\u70b9\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\u6211\u4eec\u7684 Keys \u548c\u53c2\u6570\uff0c\u5b83\u91cc\u9762\u90fd\u653e\u4e86\u4ec0\u4e48\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u8fd9\u4e2a\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u7684\u7528\u6237\u7684\u4e00\u4e2a\u9886\u5238\u7684 Key \u524d\u7f00\uff1b<\/li>\n\n\n\n<li>\u8fd9\u4e2a\u7684\u8bdd\uff0c\u5c31\u662f\u6211\u4eec\u5355\u4e2a\u4f18\u60e0\u5238\u6a21\u677f\u8981\u53bb\u9650\u5236\u7684\u6570\u91cf\uff1b<\/li>\n\n\n\n<li>\u7136\u540e\u518d\u5f80\u4e0b\u7684\u8bdd\uff0c\u5c31\u662f\u4e00\u4e2a\u6211\u4eec\u7684\u6a21\u677f ID\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u6211\u4eec Keys \u7684\u8bdd\u5c31\u6bd4\u8f83\u591a\u4e86\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u9996\u5148\u7b2c\u4e00\u70b9\uff0c\u5c31\u662f\u6211\u4eec\u672c\u6b21\u9700\u8981\u53c2\u4e0e\u7684\u7528\u6237 ID\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u8fd9\u4e2a\u5c31\u6bd4\u8f83\u6709\u610f\u601d\u4e86\u3002\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\uff0c\u6211\u4eec\u53ef\u4ee5\u770b\u4e00\u4e0b\u5b83\u662f\u600e\u4e48\u62fc\u7684\uff1a\u9996\u5148\u524d\u9762\u662f\u6211\u4eec\u7684\u6a21\u677f ID\uff0c\u7136\u540e\u8fd9\u91cc\u662f\u6211\u4eec\u7528\u6237\u7684\u9886\u5238 ID\uff08\u521a\u624d\u6211\u4eec\u65b0\u589e\u7684\u6570\u636e\u5e93\u7684\u4e3b\u952e ID\uff09\u3002<\/p>\n\n\n\n<p>\u597d\u50cf\u53d1\u73b0\u4e00\u4e0b\uff0c\u4f3c\u4e4e\u4ee3\u7801\u6709\u70b9\u591a\u4f59\u2014\u2014\u8fd9\u91cc\u6211\u5c31\u5148\u4e0d\u6539\uff0c\u56e0\u4e3a\u6211\u53d1\u73b0\u914d\u7f6e\u91cc\u9762\u5df2\u7ecf\u6709\u4f18\u60e0\u5238\u6a21\u677f ID \u4e86\uff0c\u5176\u5b9e\u6ca1\u6709\u5fc5\u8981\u62fc\u3002\u65e2\u7136\u62fc\u4e86\uff0c\u5c31\u5148\u8fd9\u4e48\u53bb\u7528\uff0c\u5bf9\u5427\uff1f\u5176\u5b9e\u8fd9\u6837\u5199\u7684\u8bdd\u4f1a\u589e\u52a0\u4e00\u4e9b\u7f51\u7edc\u7684\u6d88\u8017\u3002\u6211\u4eec\u5c31\u4e0d\u5728\u8fd9\u4e2a\u8bfe\u5802\u4e0a\u53bb\u7ed9\u5927\u5bb6\u6539\u4e86\uff0c\u5927\u5bb6\u77e5\u9053\u6709\u8fd9\u4e2a\u4e8b\u60c5\u5c31\u597d\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u4e3a\u4ec0\u4e48\u8981\u628a\u7528\u6237\u7684\u9886\u5238 ID \u548c\u6211\u4eec\u7684\u4f18\u60e0\u5238\u6a21\u677f\u62fc\u4e00\u8d77\uff1f<\/p>\n\n\n\n<p>\u6b63\u5e38\u6765\u8bb2\uff0c\u6211\u4e00\u4e2a\u7528\u6237\u7684\u9886\u5238\u91cc\u9762\uff0c\u5bf9\u5427\uff1f\u6211\u6709\u4e00\u4e2a Key \u5bf9\u5e94\u7684\u662f\u6211\u6240\u6709\u7684\u4f18\u60e0\u5238\uff0c\u8fd9\u6837\u7684\u8bdd\u5b83\u5e94\u8be5\u662f\u4e00\u4e2a\u5355\u4e2a\u7684\u503c\uff0c\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u4f46\u662f\u6211\u8fd9\u91cc\u628a\u5b83\u8003\u8651\u628a\u4f18\u60e0\u5238\u7684\u6a21\u677f\u7ed9\u5b83\u52a0\u8fdb\u53bb\uff0c\u662f\u56e0\u4e3a\u6211\u4eec\u5728\u540e\u9762\u662f\u6709\u4e00\u4e9b\u8981\u7528\u5230\u7684\u3002<\/p>\n<\/blockquote>\n\n\n\n<p><\/p>\n\n\n\n<p>\u4f17\u6240\u5468\u77e5\uff0c\u6211\u4eec\u7684\u9886\u5238\u8bb0\u5f55\u5b83\u662f\u901a\u8fc7\u4ec0\u4e48\u53bb\u8fdb\u884c\u5206\u7247\u7684\uff1f\u5b83\u662f\u901a\u8fc7\u7528\u6237 ID<\/p>\n\n\n\n<p>\u5047\u8bbe\u6211\u8981\u53bb\u67e5\u7528\u6237\u7684\u53ef\u7528\u4f18\u60e0\u5238\u5217\u8868\u7684\u65f6\u5019\uff0c\u5982\u679c\u6211\u6ca1\u6709\u4f18\u60e0\u5238\u6a21\u677f ID\uff0c\u6211\u600e\u4e48\u80fd\u77e5\u9053\u7528\u6237\u7684\u8fd9\u5f20\u5238\u80fd\u4e0d\u80fd\u88ab\u7528\u4e8e\u5546\u54c1\u6216\u8005\u8ba2\u5355\uff1f \u4e0d\u77e5\u9053<br>\u8fd9\u6837\u7684\u8bdd\uff0c\u4f60\u662f\u4e0d\u662f\u8fd8\u5f97\u53bb\u6570\u636e\u5e93\u67e5\u4e00\u904d\u2014\u2014\u6839\u636e ID \u53bb\u67e5\u4e00\u4e0b\u8bf4\u6211\u7684\u4f18\u60e0\u5238\u6a21\u677f\u662f\u4ec0\u4e48\uff0c\u518d\u53bb\u67e5\u4e00\u4e0b\u6a21\u677f\u5bf9\u5e94\u7684\u89c4\u5219\u3002\u8fd9\u6837\u7684\u8bdd\uff0c\u4f60\u662f\u4e00\u5b9a\u8981\u67e5\u4e00\u6b21\u6570\u636e\u5e93\u3002<br>\u5728\u8fd9\u91cc\uff0c\u56e0\u4e3a\u6211\u4eec\u67e5\u8be2\u7528\u6237\u53ef\u7528\u7684\u4f18\u60e0\u5238\u5217\u8868\uff0c\u6216\u8005\u8bf4\u6211\u4eec\u53bb\u6838\u9500\u4f18\u60e0\u5238\u7684\u65f6\u5019\uff0c\u4e00\u5b9a\u8981\u7528\u5230\u4f18\u60e0\u5238\u6a21\u677f\u7684\u3002\u6240\u4ee5\u8bf4\uff0c\u6211\u4eec\u662f\u4e00\u5b9a\u8981\u907f\u514d\u4ed6\u53bb\u67e5\u6570\u636e\u5e93\u7684\u3002<\/p>\n\n\n\n<p>\u6240\u4ee5\u5728\u8fd9\u91cc\u9762\uff0c\u6211\u4eec\u5c31\u628a\u5b83\u7684\u6a21\u677f\u7ed9\u5b83\u62fc\u4e00\u8d77\u3002\u8fd9\u6837\u7684\u8bdd\uff0c\u7b49\u5230\u540e\u9762\u6211\u4eec\u662f\u76f4\u63a5\u7528\u7684\uff0cOK\u3002<\/p>\n\n\n\n<p>\u7b2c\u4e09\u4e2a\u53c2\u6570\u7684\u8bdd\uff0c\u662f\u5f53\u524d\u7684\u4e00\u4e2a\u65f6\u95f4\u2014\u2014\u540e\u9762\u4f1a\u7528\u5230\u3002\u56e0\u4e3a\u6211\u4eec\u662f\u7528 ZSet \u53bb\u5b58\u7684\uff0c\u6240\u4ee5\u8bf4\u6211\u4eec\u901a\u8fc7\u5f53\u524d\u7684\u65f6\u95f4\u6765\u7ed9\u7528\u6237\u7684\u9886\u5238\u8bb0\u5f55\u6392\u4e2a\u5e8f\u3002\u518d\u5f80\u540e\u7684\u8bdd\uff0c\u5c31\u662f\u56e0\u4e3a\u6211\u4eec\u8981\u9650\u5236\u4f18\u60e0\u5238\u6a21\u677f\u5b83\u7684\u4e00\u4e2a\u9886\u5238\u7684\u6b21\u6570\uff0c\u6240\u4ee5\u8bf4\u6211\u4eec\u8981\u7ed9\u5b83\u4e00\u4e2a\u8fc7\u671f\u65f6\u95f4\u3002\u56e0\u4e3a\u4f18\u60e0\u5238\u6a21\u677f\u662f\u6709\u8fc7\u671f\u65f6\u95f4\u7684\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u6240\u4ee5\u8bf4\u6211\u4eec\u8981\u53d6\u5b83\u7684\u8fc7\u671f\u65f6\u95f4\u2014\u2014\u4ece\u73b0\u5728\u5230\u5b83\u7684\u8fc7\u671f\u65f6\u95f4\u4e00\u5171\u6709\u591a\u5c11\u79d2\uff0c\u7136\u540e\u628a\u8fd9\u4e2a\u503c\u8bbe\u7f6e\u7ed9 Redis \u7684\u53c2\u6570\u3002<code>List args = Arrays.asList(args, size = 4);<br>userIdsJson, userIdsJson: \"[\\\"184837815886552667\\\",\\\"184837815877719251\\\",\\\"1848378158684917782\\\",\\\"187<br>couponIdsJson, couponIdsJson: \"[\\\"1936786368782878978_1936786403177820160\\\",\\\"1936786368782878978_193<br>String.valueOf(new Date().getTime()),<br>\/\/ \u53d6\u5f53\u524d\u65f6\u95f4\u8ddd\u79bb\u4f18\u60e0\u5238\u6a21\u677f\u7ed3\u675f\u65f6\u95f4\u4e4b\u95f4\u7684\u79d2\u6570\u4e3a Limit Key \u7684\u8fc7\u671f\u65f6\u95f4<br>Duration.between(<br>LocalDateTime.now(),<br>event.getValidEndTime().toInstant().atZone(ZoneId.systemDefault()).toLocalDateTime()<br>).getSeconds()<br>);<\/code><\/p>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u83b7\u53d6\u4f20\u9012\u7684\u53c2\u6570\nlocal userIds = cjson.decode(ARGV&#91;1]) -- \u7528\u6237 ID \u96c6\u5408\uff0cJSON \u683c\u5f0f\u7684\u5b57\u7b26\u4e32\nlocal couponIds = cjson.decode(ARGV&#91;2]) -- \u4f18\u60e0\u5238 ID \u96c6\u5408\uff0cJSON \u683c\u5f0f\u7684\u5b57\u7b26\u4e32\nlocal userIdPrefix = KEYS&#91;1] -- \u7528\u6237 ID \u524d\u7f00\uff08\u4ece KEYS \u83b7\u53d6\uff09\nlocal limitKeyPrefix = KEYS&#91;2] -- \u7528\u6237\u4f18\u60e0\u5238\u6a21\u677f\u9650\u5236 ID\nlocal currentTemplateId = tonumber(KEYS&#91;3]) -- \u4f18\u60e0\u5238\u6a21\u677f ID\nlocal currentTime = tonumber(ARGV&#91;3]) -- \u83b7\u53d6\u5f53\u524d Unix \u65f6\u95f4\u6233\uff08\u6beb\u79d2\uff09\nlocal couponTemplateValidEndTime = tonumber(ARGV&#91;4]) -- \u4f18\u60e0\u5238\u6a21\u677f\u5230\u671f\u65f6\u95f4\n\n-- \u904d\u5386\u7528\u6237 ID \u96c6\u5408\nfor i, userId in ipairs(userIds) do\n    local key = userIdPrefix .. userId -- \u62fc\u63a5\u7528\u6237 ID \u524d\u7f00\u548c\u7528\u6237 ID\n    local couponId = couponIds&#91;i] -- \u83b7\u53d6\u5bf9\u5e94\u7684\u4f18\u60e0\u5238 ID\n    if couponId then\n        redis.call('ZADD', key, currentTime, couponId) -- \u6dfb\u52a0\u4f18\u60e0\u5238 ID \u5230 ZSet \u4e2d\n\n        local limitKey = limitKeyPrefix .. userId .. '_' .. couponTemplateId\n        redis.call('INCR', limitKey) -- \u6dfb\u52a0\u7528\u6237\u548c\u4f18\u60e0\u5238\u7684\u9886\u53d6\u6b21\u6570\uff0c\u65b9\u4fbf\u540e\u7eed\u5bf9\u7528\u6237\u8fdb\u884c\u9886\u53d6\u6b21\u6570\u524d\u7f6e\u9650\u5236\n        redis.call('EXPIRE', limitKey, couponTemplateValidEndTime) -- \u6dfb\u52a0\u7528\u6237\u4f18\u60e0\u5238\u6a21\u677f\u9650\u5236\u9886\u53d6 Key \u8fc7\u671f\u65f6\u95f4\n    end\nend<\/code><\/pre>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u5757\u64cd\u4f5c\u7684\u8bdd\uff0c\u901a\u8fc7 ZAdd \u7684\u5f62\u5f0f\uff0c\u5c06\u5f53\u524d\u7684\u201c\u4f18\u60e0\u5238\u6a21\u677f ID + \u7528\u6237\u9886\u5238 ID\u201d\u62fc\u63a5\u597d\u7684 Key \u52a0\u8fdb\u53bb\uff0c\u5b83\u7684\u503c\u5c31\u662f\u6392\u5e8f\u5b57\u6bb5\uff08\u5f53\u524d\u7684\u65f6\u95f4\uff09\uff1b<\/li>\n\n\n\n<li>\u7136\u540e\uff0c\u6211\u8981\u7ed9\u5bf9\u5e94\u7684\u9650\u5236\u7528\u6237\u9886\u4f18\u60e0\u5238\u7684\u5b57\u6bb5\uff0c\u7ed9\u5927\u5bb6\u81ea\u589e\u4e00\u4e0b\uff1b<\/li>\n\n\n\n<li>\u7136\u540e\u7ed9\u5b83\u8bbe\u7f6e\u4e00\u4e2a\u8fc7\u671f\u65f6\u95f4\u3002<\/li>\n<\/ul>\n\n\n\n<pre class=\"wp-block-code\"><code>\/\/ \u589e\u52a0\u5e93\u5b58\u56de\u6eda\u65b9\u6848\uff0c\u5982\u679c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u88ab\u6821\u9a8c\uff0c\u9700\u8981\u5c06 Redis \u9884\u6263\u51cf\u5e93\u5b58\u56de\u6eda\nint originalUserCouponSize = batchUserMaps.size();\nint availableUserCouponSize = userCouponDOLList.size();\nint rollbackStock = originalUserCouponSize - availableUserCouponSize;\nif (rollbackStock > 0) {\n    \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u6570\u91cf\n    String redisTemplateOpsForHash = String.format(EngineRedisConst.COUON_TEMPLATE_KEY, event.getCouponTemplateId()),\n            hashKey = \"stock\",\n            rollbackStock\n    );\n\n    \/\/ \u56de\u6eda\u4f18\u60e0\u5238\u6a21\u677f\u6570\u636e\u5e93\u5e93\u5b58\u6570\u91cf\n    couponTemplateMapper.incrementCouponTemplateStock(event.getShopNumber(), event.getCouponTemplateId(), rollbackStock);\n}<\/code><\/pre>\n\n\n\n<p>\u7136\u540e\u6211\u4eec\u8fd9\u8fb9\u6267\u884c\u6210\u529f\u4e4b\u540e\uff0c\u7136\u540e\u6211\u4eec\u5c31\u8981\u53bb\u589e\u52a0\u4e00\u4e2a\u5224\u65ad\u4e86\uff1a\u5982\u679c\u8bf4\u4f60\u7528\u6237\u5df2\u7ecf\u9886\u8fc7\u4f18\u60e0\u5238\uff0c\u7136\u540e\u5728\u521a\u624d\u90a3\u91cc\u88ab\u552f\u4e00\u7d22\u5f15\u6821\u9a8c\u4f4f\u4e86\uff0c\u76f8\u5f53\u4e8e\u6211\u4eec\u5df2\u7ecf\u628a\u5728\u7f13\u5b58\u548c\u6570\u636e\u5e93\u91cc\u9762\u7684\u4f18\u60e0\u5238\u7684\u5e93\u5b58\u7ed9\u6263\u51cf\u4e86\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u8fd9\u4e2a\u65f6\u5019\u4f46\u662f\u6ca1\u6709\u7528\u4e0a\uff0c\u6211\u4eec\u662f\u4e0d\u662f\u8981\u628a\u5b83\u7ed9\u56de\u6eda\u56de\u53bb\uff1f\u5bf9\u4e0d\u5bf9\uff1f<\/p>\n\n\n\n<p>\u5176\u5b9e\u672c\u6765\u50cf\u8fd9\u79cd\u80fd\u591f\u8fdb\u884c\u5206\u53d1\u7684\u90ae\u4ef6\u2014\u2014\u5176\u5b9e\u6ca1\u6709\uff0c\u5176\u5b9e\u4e0d\u9700\u8981\u56de\u6eda\u3002\u5c31\u662f\u4ece\u6211\u7684\u89d2\u5ea6\u4e0a\u6765\u770b\uff0c\u4e0d\u9700\u8981\u56de\u6eda\uff0c\u56e0\u4e3a\u5b83\u4e0d\u5c5e\u4e8e\u90a3\u79cd\u5f88\u91cd\u8981\u7684\u4f18\u60e0\u5238\u3002\u4f46\u662f\u4e3a\u4e86\u4fdd\u969c\u6211\u4eec\u4e1a\u52a1\u903b\u8f91\u7684\u5408\u7406\u6027\uff0c\u6211\u4eec\u8fd8\u662f\u7ed9\u4ed6\u8fdb\u884c\u4e86\u4e00\u6b21\u56de\u6eda\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u56de\u6eda\u7684\u8bdd\u5f88\u7b80\u5355\uff1a\u5c31\u662f\u6211\u6b63\u5e38\u7684\u903b\u8f91\u2014\u2014\u6211\u8981\u53bb\u6263\u51cf\u7684\u4e00\u4e2a\u6570\u636e\uff0c\u4ee5\u53ca\u4f60\u6700\u7ec8\u8fd4\u56de\u7ed9\u6211\u7684\u4e00\u4e2a\u7528\u6237\u7684\u9886\u5238\u96c6\u5408\u7684\u4e00\u4e2a\u8bb0\u5f55\u957f\u5ea6\u8fdb\u884c\u6bd4\u8f83\u3002\u5982\u679c\u8bf4\u5c0f\u4e8e\u6211\u4eec\u9884\u671f\u7684\u6279\u91cf\u5927\u5c0f\uff0c\u5c31\u8bc1\u660e\u6709\u88ab\u552f\u4e00\u7d22\u5f15\u6821\u9a8c\u7684\u8bb0\u5f55\u3002<\/p>\n\n\n\n<p>\u5927\u5bb6\u53ef\u4ee5\u770b\u5230\u8fd9\u4e2a\u6570\u636e\uff1a\u8fd9\u4e2a\u6570\u636e\u7684\u8bdd\uff0c\u5c31\u662f\u5982\u679c\u8bf4\u88ab\u552f\u4e00\u7d22\u5f15\u5360\u4f4f\u7684\u8bdd\uff0c\u6211\u4eec\u4f1a\u4ece\u8fd9\u91cc\u9762\u628a\u5b83\u79fb\u9664\u6389\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u5728\u8fd9\u91cc\u5220\u9664\u5df2\u7ecf\u91cd\u590d\u7684\u8bb0\u5f55\uff0c\u6240\u4ee5\u8bf4\u662f\u901a\u8fc7\u8fd9\u4e2a\u6570\u636e\u7ed9\u4ed6\u8fdb\u884c\u4e00\u4e2a\u6263\u51cf\u7684\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u56de\u6eda\u7684\u8bdd\u5f88\u7b80\u5355\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u4e2a\uff0c\u5c31\u662f\u56de\u6eda\u5b83\u7684\u6570\u636e\u5e93\uff08\u628a\u5931\u8d25\u7684\u8bb0\u5f55\u4ece\u5f85\u63d2\u5165\u96c6\u5408\u4e2d\u79fb\u9664\uff09\uff1b<\/li>\n\n\n\n<li>\u7b2c\u4e8c\u4e2a\u7684\u8bdd\uff0c\u56de\u6eda\u5b83\u7684\u5e93\u5b58\uff08\u628a\u591a\u6263\u7684\u5e93\u5b58\u52a0\u56de\u53bb\uff09\u3002<\/li>\n<\/ul>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"4\">4. \u4fee\u6539\u4f18\u60e0\u5238\u5206\u53d1\u8bb0\u5f55\u72b6\u6001<\/h3>\n\n\n\n<p>\u4e0a\u9762\u8bb2\u4e86\u5230\u8fbe\u6279\u91cf\u6267\u884c\u6761\u4ef6\u7684\u6d88\u606f\uff0c\u7136\u540e\u6211\u4eec\u63a5\u7740\u8bf4 Excel \u8bfb\u53d6\u5b8c\u6267\u884c\u7684\u6d88\u606f\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> @Component\n @RequiredArgsConstructor\n @RocketMQMessageListener(\n         topic = \"one-coupon_distribution-service_coupon-execute-distribution_topic${unique-name:}\",\n         consumerGroup = \"one-coupon_distribution-service_coupon-execute-distribution_cg${unique-name:}\"\n )\n @Slf4j(topic = \"CouponExecuteDistributionConsumer\")\n public class CouponExecuteDistributionConsumer implements RocketMQListener&lt;MessageWrapper&lt;CouponTemplateDistributionEvent&gt;&gt; {\n \u200b\n     private final UserCouponMapper userCouponMapper;\n     private final CouponTemplateMapper couponTemplateMapper;\n     private final CouponTaskMapper couponTaskMapper;\n     private final CouponTaskFailMapper couponTaskFailMapper;\n     private final StringRedisTemplate stringRedisTemplate;\n \u200b\n     @Lazy\n     @Autowired\n     private CouponExecuteDistributionConsumer couponExecuteDistributionConsumer;\n \u200b\n     private static final String BATCH_SAVE_USER_COUPON_LUA_PATH = \"lua\/batch_user_coupon_list.lua\";\n \u200b\n     @Transactional(rollbackFor = Exception.class)\n     @Override\n     public void onMessage(MessageWrapper&lt;CouponTemplateDistributionEvent&gt; messageWrapper) {\n         \/\/ ......\n         \/\/ \u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6\u4e3a TRUE\uff0c\u4ee3\u8868\u5df2\u7ecf\u6ca1\u6709 Excel \u8bb0\u5f55\u4e86\n         if (event.getDistributionEndFlag()) {\n             String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n             Long batchUserIdsSize = stringRedisTemplate.opsForSet().size(batchUserSetKey);\n             event.setBatchUserSetSize(batchUserIdsSize.intValue());\n \u200b\n             decrementCouponTemplateStockAndSaveUserCouponList(event);\n             List&lt;String&gt; batchUserMaps = stringRedisTemplate.opsForSet().pop(batchUserSetKey, Integer.MAX_VALUE);\n             \/\/ \u6b64\u65f6\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\u5982\u679c\u8fd8\u6709\u503c\uff0c\u5c31\u610f\u5473\u7740\u53ef\u80fd\u5e93\u5b58\u4e0d\u8db3\u5f15\u8d77\u7684\n             if (CollUtil.isNotEmpty(batchUserMaps)) {\n                 \/\/ ...... \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\n             }\n \u200b\n             \/\/ \u786e\u4fdd\u6240\u6709\u7528\u6237\u90fd\u5df2\u7ecf\u63a5\u5230\u4f18\u60e0\u5238\u540e\uff0c\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u5b8c\u6210\u65f6\u95f4\n             CouponTaskDO couponTaskDO = CouponTaskDO.builder()\n                     .id(event.getCouponTaskId())\n                     .status(CouponTaskStatusEnum.SUCCESS.getStatus())\n                     .completionTime(new Date())\n                     .build();\n             couponTaskMapper.updateById(couponTaskDO);\n         }\n     }\n }<\/code><\/pre>\n\n\n\n<p>\u548c\u4e0a\u9762\u6d41\u7a0b\u4e00\u81f4\uff0c\u9996\u5148\u6263\u51cf MySQL \u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\uff0c\u7136\u540e\u4fdd\u5b58\u7528\u6237 Redis \u9886\u5238\u8bb0\u5f55\u7b49\u64cd\u4f5c\u3002\u6700\u7ec8\u6240\u6709\u903b\u8f91\u6267\u884c\u5b8c\u540e\uff0c\u6211\u4eec\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u7684\u72b6\u6001\u548c\u5b8c\u6210\u65f6\u95f4\u5373\u53ef\u3002<\/p>\n\n\n\n<p>\u5982\u679c\u8bf4\u6211\u7684\u5206\u53d1\u4efb\u52a1\u5df2\u7ecf\u7ed3\u675f\u4e86\uff0c\u90a3\u8fd9\u4e2a\u65f6\u5019\u6211\u8981\u505a\u54ea\u4e9b\u4e8b\u60c5\uff1f<\/p>\n\n\n\n<p>\u7b2c\u4e00\u6b65\u7684\u8bdd,\u6211\u83b7\u53d6\u5230\u521a\u624d\u6211\u6700\u540e\u7684\u4e00\u4e2a\u6570\u636e\u91cf\uff0c\u4e5f\u5c31\u662f\u6211\u4eec\u5bf9\u5e94\u7684\u7f13\u51b2\u961f\u5217\u5b83\u91cc\u9762\u8fd8\u6709\u591a\u5c11\u5143\u7d20\uff0c\u7136\u540e\u6211\u4eec\u53bb\u7ed9\u4ed6\u8bbe\u7f6e\u5230\u6211\u4eec\u7684\u6279\u91cf\u7684\u4e00\u4e2a\u5927\u5c0f\u91cc\u9762\uff0c\u7136\u540e\u518d\u7ed9\u5b83\u8fdb\u884c\u4e00\u6b21\u521a\u624d\u4e0a\u9762\u7684\u903b\u8f91\u3002<\/p>\n\n\n\n<p>\u53ef\u4ee5\u770b\u5230\u8fd9\u4e2a\u903b\u8f91\uff1a\u4ed6\u5f39\u4e86\u3001\u64cd\u4f5c\u5b8c\u4e4b\u540e\uff0c\u5982\u679c\u8bf4\u91cc\u9762\u8fd8\u6709\u5bf9\u5e94\u7684\u8bb0\u5f55\uff0c\u57fa\u672c\u4e0a\u53ef\u4ee5\u65ad\u5b9a\u6211\u4eec\u7684\u4f18\u60e0\u5238\u5e93\u5b58\u91cc\u9762\u5df2\u7ecf\u6ca1\u6709\u503c\u4e86\u3002\u5230\u8fd9\u91cc\u7684\u8bdd\uff0c\u6211\u4eec\u5c31\u8981\u628a\u8fd9\u4e9b\u8bb0\u5f55\u7ed9\u5b83\u65b0\u589e\u5230\u6211\u4eec\u5bf9\u5e94\u7684\u4f18\u60e0\u5238\u4efb\u52a1\u5931\u8d25\u7684\u8bb0\u5f55\u91cc\u9762\u53bb\u3002<\/p>\n\n\n\n<p>couponTaskFailMapper.insert(couponTaskFailDoList);<br>\u6700\u540e\u7684\u8bdd\uff0c\u7136\u540e\u6211\u4eec\u628a\u4e4b\u524d\u4e0a\u4e00\u8282\u8bfe\u91cc\u9762\u2014\u2014\u5728 Excel \u89e3\u6790\u5b8c\u7684\u53bb\u4fee\u6539\u6211\u4eec\u5206\u53d1\u4efb\u52a1\u7684\u903b\u8f91\u2014\u2014\u7ed9\u52a0\u5230\u4e86\u8fd9\u91cc\uff0c\u7136\u540e\u7ed9\u5b83\u53d8\u66f4\u4e00\u4e0b\u5b83\u7684\u4e00\u4e2a\u72b6\u6001\u4ee5\u53ca\u5b83\u7684\u4e00\u4e2a\u5b8c\u6210\u65f6\u95f4\u3002<br>\/\/\u786e\u4fdd\u6240\u6709\u7528\u6237\u90fd\u5df2\u7ecf\u63a5\u5230\u4f18\u60e0\u5238\u540e\uff0c\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u5b8c\u6210\u65f6\u95f4<br>CouponTaskDO couponTaskDO = CouponTaskDO.builder()<br>.id\uff08event.getCouponTaskId())<br>.status(CouponTaskStatusEnum.SUcCESS.getStatus())<br>.completionTime(new Date(\uff09)<br>.build(\uff09;<br>couponTaskMapper.updateById(couponTaskDO);<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u4fdd\u5b58\u7528\u6237\u9886\u5238\u5931\u8d25\u8bb0\u5f55<\/h2>\n\n\n\n<p>\u5982\u679c\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u8fc7\u8be5\u4f18\u60e0\u5238\u6216\u8005\u5e93\u5b58\u4e0d\u8db3\uff0c\u4f1a\u88ab\u552f\u4e00\u7d22\u5f15\u6821\u9a8c\u4f4f\uff0c\u90a3\u6211\u4eec\u5c31\u9700\u8981\u5c06\u5bf9\u5e94\u7684\u62a5\u9519\u8bb0\u5f55\u6dfb\u52a0\u5230 <code>t_coupon_task_fail<\/code> \u8868\u4e2d\u3002<\/p>\n\n\n\n<pre class=\"wp-block-code\"><code> \u200b@Transactional(rollbackFor = Exception.class)\n @Override\n public void onMessage(MessageWrapper&lt;CouponTemplateDistributionEvent&gt; messageWrapper) {\n     \/\/ ......\n     \/\/ \u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6\u4e3a TRUE\uff0c\u4ee3\u8868\u5df2\u7ecf\u6ca1\u6709 Excel \u8bb0\u5f55\u4e86\n     if (event.getDistributionEndFlag()) {\n         String batchUserSetKey = String.format(DistributionRedisConstant.TEMPLATE_TASK_EXECUTE_BATCH_USER_KEY, event.getCouponTaskId());\n         Long batchUserIdsSize = stringRedisTemplate.opsForSet().size(batchUserSetKey);\n         event.setBatchUserSetSize(batchUserIdsSize.intValue());\n \u200b\n         decrementCouponTemplateStockAndSaveUserCouponList(event);\n         List&lt;String&gt; batchUserMaps = stringRedisTemplate.opsForSet().pop(batchUserSetKey, Integer.MAX_VALUE);\n         \/\/ \u6b64\u65f6\u5f85\u4fdd\u5b58\u5165\u5e93\u7528\u6237\u4f18\u60e0\u5238\u5217\u8868\u5982\u679c\u8fd8\u6709\u503c\uff0c\u5c31\u610f\u5473\u7740\u53ef\u80fd\u5e93\u5b58\u4e0d\u8db3\u5f15\u8d77\u7684\n         if (CollUtil.isNotEmpty(batchUserMaps)) {\n             \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n             List&lt;CouponTaskFailDO&gt; couponTaskFailDOList = new ArrayList&lt;&gt;(batchUserMaps.size());\n             for (String batchUserMapStr : batchUserMaps) {\n                 Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n                         .put(\"rowNum\", JSON.parseObject(batchUserMapStr).get(\"rowNum\"))\n                         .put(\"cause\", \"\u4f18\u60e0\u5238\u6a21\u677f\u5e93\u5b58\u4e0d\u8db3\")\n                         .build();\n                 CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n                         .batchId(event.getCouponTaskBatchId())\n                         .jsonObject(com.alibaba.fastjson.JSON.toJSONString(objectMap))\n                         .build();\n                 couponTaskFailDOList.add(couponTaskFailDO);\n             }\n \u200b\n             \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\n             couponTaskFailMapper.insert(couponTaskFailDOList);\n         }\n \u200b\n         \/\/ \u786e\u4fdd\u6240\u6709\u7528\u6237\u90fd\u5df2\u7ecf\u63a5\u5230\u4f18\u60e0\u5238\u540e\uff0c\u8bbe\u7f6e\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\u5b8c\u6210\u65f6\u95f4\n         CouponTaskDO couponTaskDO = CouponTaskDO.builder()\n                 .id(event.getCouponTaskId())\n                 .status(CouponTaskStatusEnum.SUCCESS.getStatus())\n                 .completionTime(new Date())\n                 .build();\n         couponTaskMapper.updateById(couponTaskDO);\n     }\n }\n \u200b\n private void batchSaveUserCouponList(Long couponTemplateId, Long couponTaskBatchId, List&lt;UserCouponDO&gt; userCouponDOList) {\n     \/\/ MyBatis-Plus \u6279\u91cf\u6267\u884c\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\n     try {\n         userCouponMapper.insert(userCouponDOList, userCouponDOList.size());\n     } catch (Exception ex) {\n         Throwable cause = ex.getCause();\n         if (cause instanceof BatchExecutorException) {\n             \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n             List&lt;CouponTaskFailDO&gt; couponTaskFailDOList = new ArrayList&lt;&gt;();\n             List&lt;UserCouponDO&gt; toRemove = new ArrayList&lt;&gt;();\n \u200b\n             \/\/ \u8c03\u7528\u6279\u91cf\u65b0\u589e\u5931\u8d25\u540e\uff0c\u4e3a\u4e86\u907f\u514d\u5927\u91cf\u91cd\u590d\u5931\u8d25\uff0c\u6211\u4eec\u901a\u8fc7\u65b0\u589e\u5355\u6761\u8bb0\u5f55\u65b9\u5f0f\u6267\u884c\n             userCouponDOList.forEach(each -&gt; {\n                 try {\n                     userCouponMapper.insert(each);\n                 } catch (Exception ignored) {\n                     Boolean hasReceived = couponExecuteDistributionConsumer.hasUserReceivedCoupon(couponTemplateId, each.getUserId());\n                     if (hasReceived) {\n                         \/\/ \u6dfb\u52a0\u5230 t_coupon_task_fail \u5e76\u6807\u8bb0\u9519\u8bef\u539f\u56e0\uff0c\u65b9\u4fbf\u540e\u7eed\u67e5\u770b\u672a\u6210\u529f\u53d1\u9001\u7684\u539f\u56e0\u548c\u8bb0\u5f55\n                         Map&lt;Object, Object&gt; objectMap = MapUtil.builder()\n                                 .put(\"rowNum\", each.getRowNum())\n                                 .put(\"cause\", \"\u7528\u6237\u5df2\u9886\u53d6\u8be5\u4f18\u60e0\u5238\")\n                                 .build();\n                         CouponTaskFailDO couponTaskFailDO = CouponTaskFailDO.builder()\n                                 .batchId(couponTaskBatchId)\n                                 .jsonObject(com.alibaba.fastjson.JSON.toJSONString(objectMap))\n                                 .build();\n                         couponTaskFailDOList.add(couponTaskFailDO);\n \u200b\n                         \/\/ \u4ece userCouponDOList \u4e2d\u5220\u9664\u5df2\u7ecf\u5b58\u5728\u7684\u8bb0\u5f55\n                         toRemove.add(each);\n                     }\n                 }\n             });\n \u200b\n             \/\/ \u6279\u91cf\u65b0\u589e t_coupon_task_fail \u8868\n             couponTaskFailMapper.insert(couponTaskFailDOList, couponTaskFailDOList.size());\n \u200b\n             \/\/ \u5220\u9664\u5df2\u7ecf\u91cd\u590d\u7684\u5185\u5bb9\n             userCouponDOList.removeAll(toRemove);\n             return;\n         }\n \u200b\n         throw ex;\n     }\n }\n \u200b\n \/**\n  * \u67e5\u8be2\u7528\u6237\u662f\u5426\u5df2\u7ecf\u9886\u53d6\u8fc7\u4f18\u60e0\u5238\n  *\n  * @param couponTemplateId \u4f18\u60e0\u5238\u6a21\u677f ID\n  * @param userId           \u7528\u6237 ID\n  * @return \u7528\u6237\u4f18\u60e0\u5238\u6a21\u677f\u9886\u53d6\u4fe1\u606f\u662f\u5426\u5df2\u5b58\u5728\n  *\/\n @Transactional(propagation = Propagation.NOT_SUPPORTED, readOnly = true)\n public Boolean hasUserReceivedCoupon(Long couponTemplateId, Long userId) {\n     LambdaQueryWrapper&lt;UserCouponDO&gt; queryWrapper = Wrappers.lambdaQuery(UserCouponDO.class)\n             .eq(UserCouponDO::getUserId, userId)\n             .eq(UserCouponDO::getCouponTemplateId, couponTemplateId);\n     return userCouponMapper.selectOne(queryWrapper) != null;\n }<\/code><\/pre>\n\n\n\n<p>Q\uff1a\u4e3a\u4ec0\u4e48\u8981\u662f\u7528\u975e\u4e8b\u52a1\u7684\u67e5\u8be2\u65b9\u5f0f\u67e5\u8be2\u5df2\u5b58\u5728\u7684\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\uff1f<\/p>\n\n\n\n<p>A\uff1a\u5982\u679c\u7b2c\u4e00\u6b21\u6279\u91cf\u4fdd\u5b58\u6570\u636e\u5e93\u5931\u8d25\u540e\uff0c\u6279\u91cf\u4fdd\u5b58\u7684\u4e00\u4e9b\u8bb0\u5f55\u4f1a\u5728\u4e8b\u52a1\u7f13\u51b2\u533a\u4e2d\uff0c\u5982\u679c\u5728\u540c\u4e00\u4e2a\u4e8b\u52a1\u4e0b\uff0c\u4f1a\u67e5\u5230\u6570\u636e\u5e93\u4e2d\u672c\u4e0d\u5b58\u5728\u7684\u8bb0\u5f55\u3002\u6211\u4eec\u4fdd\u5b58\u7528\u6237\u5df2\u9886\u53d6\u4f18\u60e0\u5238\u7684\u5931\u8d25\u8bb0\u5f55\uff0c\u9700\u8981\u662f\u6570\u636e\u5e93\u4e2d\u771f\u5b9e\u5b58\u5728\u7684\uff0c\u6240\u4ee5\u4e0d\u4f7f\u7528\u4e8b\u52a1\u7684\u65b9\u5f0f\u67e5\u8be2\u3002<\/p>\n\n\n\n<h2 class=\"wp-block-heading\">\u5e38\u89c1\u95ee\u9898\u7b54\u7591<\/h2>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"1-coupon-execute-distribution-consumer\">1. \u4e3a\u4ec0\u4e48 CouponExecuteDistributionConsumer \u6d88\u8d39\u65f6\u4f1a\u6bd4\u6279\u91cf\u7ed3\u679c\u591a\u51e0\u6761\uff1f<\/h3>\n\n\n\n<p>\u7ed9\u5927\u5bb6\u652f\u4e2a\u62db\uff0c5000 \u6279\u6b21\u662f\u751f\u4ea7\u4e2d\u7684\u53c2\u6570\uff0c\u6211\u4eec\u6d4b\u8bd5\u65f6\u5019 10 \u6761\u4e00\u4e2a\u6279\u6b21\u5c31\u597d\u4e86\uff0c\u65b9\u4fbf Debug\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u6709\u540c\u5b66\u63d0\u95ee\uff1a\u4e3a\u4ec0\u4e48 CouponExecuteDistributionConsumer \u6d88\u8d39\u65f6\u83b7\u53d6 Redis Set \u7528\u6237\u9886\u5238\u8bb0\u5f55\u4f1a\u6bd4\u6279\u91cf\u7ed3\u679c\u591a\u51e0\u6761\uff1f\u6709\u65f6\u5019\u662f 10 \u6761\uff0c\u6709\u65f6\u5019\u662f 11 \u6761\u8fd9\u6837\u3002<\/p>\n\n\n\n<p>\u56e0\u4e3a\u89e6\u53d1\u6279\u6b21\u6761\u6570\u540e\uff0c\u5f00\u59cb\u5f80 CouponExecuteDistributionConsumer \u53d1\u9001\u6d88\u606f\uff0c\u4f46\u662f\u6709\u53ef\u80fd\u5728\u6267\u884c\u6d88\u8d39\u524d\uff0cExcel \u76d1\u542c\u5668\u53c8\u5f80 Redis Set \u7ed3\u6784\u6dfb\u52a0\u4e86\u4e00\u6761\u7528\u6237\u9886\u5238\u8bb0\u5f55\uff0c\u6240\u4ee5\u5c31\u4f1a\u51fa\u73b0\u6bd4\u9884\u671f\u7684\u7ed3\u679c\u591a\u51fa\u51e0\u6761\u7684\u53ef\u80fd\u3002\u8fd9\u4e2a\u5e76\u4e0d\u4f1a\u5f71\u54cd\u7cfb\u7edf\u8fd0\u884c\uff0c\u5ffd\u7565\u5c31\u597d\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5c31\u662f\u4f60\u518d\u53bb\u8fd9\u91cc\u9762\u8fdb\u884c\u6d88\u8d39\u7684\u65f6\u5019\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u4f60\u80af\u5b9a\u4f1a\u6253 debug\uff0c\u5bf9\u4e0d\u5bf9\uff1f\u4f60\u6253\u65ad\u70b9\u7684\u65f6\u5019\uff0c\u8fd9\u91cc\u9762\u5b83\u662f\u6e90\u6e90\u4e0d\u65ad\u5730\u5f80\u4f60\u7684\u7f13\u51b2\u961f\u5217\u91cc\u9762\u4f1a\u53bb\u8bbe\u7f6e\u5b83\u7684\u4e00\u4e2a\u65b0\u589e\u503c\u7684\u3002\u660e\u767d\u5427\uff1f\u6240\u4ee5\u8bf4\u4f60\u6253\u65ad\u70b9\u662f\u4e00\u5b9a\u4f1a\u51fa\u73b0\u8fd9\u79cd\u60c5\u51b5\u7684\u3002\u6211\u4eec\u8bbe\u7f6e\u4e86\u7f13\u51b2\u7684\u4e00\u6b21\u6279\u6b21\u662f 10\uff0c\u4f46\u662f\u4f60\u53bb\u770b\u5b83\u5bf9\u5e94\u7684 Set \u91cc\u9762\u7684\u8bdd\uff0c\u5b83\u4f1a\u591a\u4e8e 10\u2014\u2014\u8fd9\u4e2a\u662f\u6b63\u5e38\u7684\u4e00\u4e2a\u73b0\u8c61\u3002<\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"2\">2. \u6d88\u606f\u961f\u5217\u662f\u5426\u9700\u8981\u987a\u5e8f\u6d88\u606f\uff1f<\/h3>\n\n\n\n<p>\u9700\u8981\u7684\u3002\u6211\u4eec\u5728\u6267\u884c\u6b63\u5f0f\u5206\u53d1\u7528\u6237\u4f18\u60e0\u5238\u7684\u6d88\u8d39\u8005\u65f6\uff0c\u9700\u8981\u4f7f\u7528\u987a\u5e8f\u6d88\u606f\u6267\u884c\u3002\u56e0\u4e3a\u6709\u53ef\u80fd\u6211\u4eec Excel \u5df2\u7ecf\u8bfb\u53d6\u5b8c\u4e86\uff0c\u5f00\u59cb\u6267\u884c <code>distributionEndFlag<\/code> \u7b49\u4e8e True \u7684\u903b\u8f91\u65f6\uff0c\u4e0a\u4e00\u4e2a 5000 \u6279\u6b21\u7684\u6d88\u606f\u8fd8\u6ca1\u6709\u6267\u884c\u5b8c\u7684\u53ef\u80fd\u3002\u5047\u8bbe 5000 \u6279\u6b21\u6d88\u606f\u91cc\u6709\u51fa\u73b0\u5f02\u5e38\uff0c\u4f46\u662f\u56e0\u4e3a <code>distributionEndFlag<\/code> \u7b49\u4e8e True \u7684\u903b\u8f91\u5148\u6267\u884c\uff0c\u5c31\u4f1a\u5b58\u5728\u8bb0\u5f55\u4e0d\u4e0a\u7684\u573a\u666f\u3002<\/p>\n\n\n\n<p>\u4e3a\u6b64\uff0c\u6211\u4eec\u9700\u8981\u53d1\u9001\u987a\u5e8f\u6d88\u606f\uff0c\u6709\u4e24\u79cd\u65b9\u5f0f\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>Topic \u4e0b\u5355\u961f\u5217\uff1a\u8fd9\u6837\u80fd\u4fdd\u8bc1\u987a\u5e8f\uff0c\u4f46\u662f\u7f3a\u70b9\u662f\u5e76\u53d1\u592a\u6162\u3002<\/li>\n\n\n\n<li>Topic \u4e0b\u591a\u961f\u5217\uff0c\u6839\u636e\u5206\u53d1\u4efb\u52a1 ID \u8fdb\u884c Hash \u6295\u9012\u5230\u56fa\u5b9a\u961f\u5217\uff1a\u5e76\u53d1\u8f83\u9ad8\uff0c\u63a8\u8350\u4f7f\u7528\u8be5\u7b56\u7565\u3002<\/li>\n<\/ul>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u53c2\u8003\u6587\u7ae0\uff1a<a href=\"https:\/\/rocketmq.apache.org\/zh\/docs\/featureBehavior\/03fifomessage\" target=\"_blank\"  rel=\"nofollow\" >RocketMQ \u987a\u5e8f\u6d88\u606f<\/a><\/p>\n<\/blockquote>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u9996\u5148\u662f\u9700\u8981\u7684\u3002\u56e0\u4e3a\u6211\u4eec\u53bb\u8fdb\u884c\u4e00\u4e2a\u5206\u53d1\u6d88\u8d39\u7684\u65f6\u5019\uff0c\u6709\u4e00\u79cd\u53ef\u80fd\uff1a\u6211\u5728\u6267\u884c\u7684\u65f6\u5019\uff0c\u5047\u5982\u8bf4\u6211\u7684\u4e00\u4e2a\u6279\u6b21\u5230\u4e86\uff0c\u6211\u53bb\u53d1\u9001\u4e86\u4e2a\u6d88\u606f\u3002\u4f46\u662f\u6070\u5de7\u6211\u7684\u4e0b\u4e00\u6761\u2014\u2014\u6211\u7684 Excel \u5df2\u7ecf\u6ca1\u8bb0\u5f55\u4e86\uff0c\u90a3\u4e48\u6211\u4f1a\u628a\u8fd9\u4e2a\u4e5f\u53bb\u53d1\u9001\u5230\u5bf9\u5e94\u7684\u6d88\u606f\u961f\u5217\u91cc\u9762\u3002\u7136\u540e\u76f8\u5f53\u4e8e\u8fd9\u4e24\u6761\u6d88\u606f\u662f\u540c\u65f6\u6267\u884c\u7684\u3002<\/p>\n\n\n\n<p>\u7136\u540e\u56e0\u4e3a\u8fd9\u91cc\u9762\u7684\u903b\u8f91\u4f1a\u6267\u884c\u5f97\u7a0d\u5fae\u590d\u6742\u4e00\u70b9\u2014\u2014\u7b2c\u4e00\u4e2a\u6d88\u606f\u4f1a\u7a0d\u5fae\u590d\u6742\u4e00\u70b9\uff0c\u7b2c\u4e8c\u4e2a\u6d88\u606f\u7684\u8bdd\u5b83\u53ef\u80fd\u4f1a\u66f4\u5feb\u4e00\u70b9\u3002\u4e5f\u5c31\u610f\u5473\u7740\u6211\u4eec\u8fd9\u91cc\u9762\u8bbe\u7f6e\u4e86\u5b83\u5df2\u7ecf\u6267\u884c\u5b8c\u6210\u4e86\uff0c\u4f46\u662f\u5b83\u7684\u6d88\u606f\u5176\u5b9e\u8fd8\u5728\u6267\u884c\u7684\u8fc7\u7a0b\u5f53\u4e2d\u3002<\/p>\n\n\n\n<p>\u6240\u4ee5\u8bf4\uff0c\u6211\u662f\u63a8\u8350\u5927\u5bb6\u7528\u987a\u5e8f\u6d88\u606f\u7684\u3002<\/p>\n\n\n\n<p>\u90a3\u987a\u5e8f\u6d88\u606f\u7684\u8bdd\u6709\u4e24\u79cd\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u7b2c\u4e00\u79cd\u7684\u8bdd\uff0c\u6211\u4eec\u5728\u4e00\u4e2a Topic \u4e0b\u9762\u4f7f\u7528\u5355\u4e2a\u961f\u5217\uff0c\u8fd9\u6837\u7684\u8bdd\u80fd\u591f\u4fdd\u8bc1\u6211\u4eec\u8fd9\u4e2a\u6d88\u606f\u7684\u987a\u5e8f\u6027\uff0c\u4f46\u662f\u7f3a\u70b9\u5c31\u662f\u5e76\u53d1\u4f1a\u6162\u4e00\u70b9\uff0c\u5bf9\u5427\uff1f<\/li>\n\n\n\n<li>\u7b2c\u4e8c\u4e2a\u5c31\u662f\u6211\u4eec\u4e00\u4e2a Topic \u4e0b\u9762\u591a\u961f\u5217\uff0c\u6211\u4eec\u53ef\u4ee5\u6839\u636e\u6211\u4eec\u5206\u53d1\u4efb\u52a1 ID\uff0c\u901a\u8fc7 Hash \u7684\u5f62\u5f0f\u653e\u5230\u56fa\u5b9a\u7684\u961f\u5217\u91cc\u9762\uff0c\u8fd9\u6837\u7684\u8bdd\u5e76\u53d1\u8fd8\u662f\u6bd4\u8f83\u9ad8\u7684\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5927\u5bb6\u53ef\u4ee5\u53bb\u770b\u4e00\u4e0b\u6211\u4eec\u7684\u53c2\u8003\u6587\u7ae0\u91cc\u9762\uff0c\u8fd9\u91cc\u9762\u662f\u6709\u5bf9\u8fd9\u79cd\u6d88\u606f\u8fdb\u884c\u4e00\u4e2a\u6f14\u793a\u793a\u4f8b\u7684\u3002<\/p>\n<\/blockquote>\n\n\n\n<h3 class=\"wp-block-heading\" id=\"3\">3. \u4e3a\u4ec0\u4e48\u8981\u62c6\u5206\u6d88\u606f\u961f\u5217\u6d88\u606f\uff1f<\/h3>\n\n\n\n<p>\u6709\u4f4d\u540c\u5b66\u63d0\u51fa\u7591\u95ee\uff0c\u4e3a\u4ec0\u4e48\u8981\u5bf9\u6d88\u606f\u961f\u5217\u8fdb\u884c\u62c6\u5206\u6d88\u606f\u6d88\u8d39\uff1f\u4ee5\u4e0b\u662f\u539f\u8bdd\uff1a<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u5728\u4f18\u60e0\u5238\u5206\u53d1\u7684\u65f6\u5019\uff0c\u6211\u4eec\u4f7f\u7528\u4e86\u6d88\u606f\u961f\u5217\u6d88\u8d39\u62c6\u5206\u65b9\u6848\uff0c\u76ee\u7684\u662f\u4f7f\u5f97 rocketmq \u7684 broker \u4e0d\u8981\u8ba4\u4e3a\u6d88\u606f\u6d88\u8d39\u8d85\u65f6\u800c\u91cd\u65b0\u6295\u9012\u3002 \u4f46\u662f\u6211\u53ef\u4ee5\u628a rocketmq \u7684\u6d88\u606f\u6d88\u8d39\u8d85\u65f6\u65f6\u95f4\u8bbe\u7f6e\u7684\u957f\u4e00\u70b9\u4e0d\u5c31\u53ef\u4ee5\u4e86\u5417\u3002\u7f51\u4e0a\u8bf4\u53ef\u4ee5\u901a\u8fc7 transactionTimeout \u53c2\u6570\u6765\u914d\u7f6e\u3002 \u800c\u4e14\u6211\u4e5f\u4e0d\u89c9\u5f97\u4e0d\u62c6\u5206\u4f1a\u4f7f\u5f97\u6d88\u8d39\u65f6\u95f4\u53d8\u5f97\u592a\u957f\uff0c\u56e0\u4e3a\u767e\u4e07\u4f18\u60e0\u5238\u53d1\u653e\u5176\u5b9e\u53ea\u9700\u8981 200s\uff0c\u6211\u4eec\u5df2\u7ecf\u901a\u8fc7\u6279\u5904\u7406\u6765\u5927\u5927\u7f29\u77ed\u65f6\u95f4\u4e86\uff0c\u90a3\u8fd9\u4e2a\u62c6\u5206\u662f\u4e0d\u662f\u663e\u5f97\u6ca1\u4ec0\u4e48\u5fc5\u8981\u5462\u3002<\/p>\n<\/blockquote>\n\n\n\n<p>\u4ece\u6211\u4eec\u7684\u573a\u666f\u6765\u770b\uff0c\u6211\u6ca1\u6709\u53d1\u73b0\u660e\u663e\u7684\u95ee\u9898\u3002\u4f46\u4e3a\u4ec0\u4e48\u4ecd\u7136\u8981\u505a\u62c6\u5206\u5462\uff1f\u5176\u5b9e\u6838\u5fc3\u76ee\u7684\u662f\u6811\u7acb\u89c4\u7ea6\u610f\u8bc6\u2014\u2014\u6d88\u606f\u961f\u5217\u7684\u6d88\u606f\u5904\u7406\u5e94\u8be5\u5c3d\u91cf\u8f7b\u91cf\uff0c\u4e0d\u9002\u5408\u5d4c\u5165\u8fc7\u957f\u7684\u4e1a\u52a1\u903b\u8f91\u3002<\/p>\n\n\n\n<p>\u5728\u672c\u5730\u6d4b\u8bd5\u65f6\uff0c\u5373\u4f7f\u5355\u6b21\u6d88\u8d39\u65f6\u95f4\u8f83\u957f\u53ef\u80fd\u4e5f\u4e0d\u4f1a\u51fa\u73b0\u95ee\u9898\uff0c\u4f46\u751f\u4ea7\u73af\u5883\u66f4\u590d\u6742\uff0c\u8fd9\u79cd\u505a\u6cd5\u53ef\u80fd\u4f1a\u5e26\u6765\u6f5c\u5728\u98ce\u9669\u3002<\/p>\n\n\n\n<p>\u603b\u7ed3\u6765\u770b\uff0c\u62c6\u5206\u6d88\u606f\u4e00\u5b9a\u4e0d\u4f1a\u51fa\u9519\uff0c\u5e76\u4e14\u6ca1\u6709\u660e\u663e\u7f3a\u9677\uff0c\u800c\u957f\u6d88\u606f\u867d\u7136\u5b9e\u73b0\u66f4\u7b80\u5355\uff0c\u4f46\u6f5c\u5728\u98ce\u9669\u4e0d\u786e\u5b9a\u3002\u5982\u679c\u662f\u6211\uff0c\u4e00\u5b9a\u4f1a\u9009\u62e9\u62c6\u5206\u3002<\/p>\n\n\n\n<blockquote class=\"wp-block-quote is-layout-flow wp-block-quote-is-layout-flow\">\n<p>\u53c2\u8003\u539f\u56de\u7b54\uff1a<a href=\"https:\/\/t.zsxq.com\/IdYVS\" target=\"_blank\"  rel=\"nofollow\" >https:\/\/t.zsxq.com\/IdYVS<\/a><\/p>\n<\/blockquote>\n\n\n\n<h2 class=\"wp-block-heading\">\u6587\u672b\u603b\u7ed3<\/h2>\n\n\n\n<p>\u672c\u7ae0\u8282\u8f83\u4e3a\u590d\u6742\uff0c\u4ee3\u7801\u903b\u8f91\u4e00\u73af\u5957\u4e00\u73af\uff0c\u4e0d\u8fc7\uff0c\u867d\u7136\u903b\u8f91\u76f8\u8f83\u4e8e v1 \u7248\u672c\u53d8\u7684\u6666\u6da9\u96be\u61c2\uff0c\u4f46\u662f\u6027\u80fd\u63d0\u5347\u975e\u5e38\u660e\u663e\u3002\u76f8\u540c\u7684 5000 \u6761\u8bb0\u5f55\uff0cv1 \u7248\u672c\u9700\u8981\u6267\u884c 1 \u5206\u949f\u5de6\u53f3\uff0cv2 \u4e5f\u5c31\u662f\u5f53\u524d\u7248\u672c\u4ec5\u9700\u8981 1 \u79d2\uff0c\u63a5\u8fd1 60 \u500d\u7684\u6027\u80fd\u63d0\u5347\u3002<\/p>\n\n\n\n<p>\u867d\u7136\u903b\u8f91\u5df2\u7ecf\u76f8\u5bf9\u5b8c\u5584\u4e86\uff0c\u4f46\u662f\u8fd8\u6709\u4e24\u4e2a\u53ef\u80fd\u5b58\u5728\u7455\u75b5\u7684\u5730\u65b9\uff0c\u5982\u4e0b\u6240\u793a\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u5982\u679c <code>batchSaveUserCouponList<\/code> \u6267\u884c\u65f6\u6570\u636e\u5e93\u5b95\u673a\uff0c\u90a3\u4e48\u5c31\u4f1a\u9762\u4e34\u4e8b\u52a1\u56de\u6eda\u95ee\u9898\uff0c\u610f\u5473\u7740\u6211\u4eec\u9700\u8981\u5c06\u4ece Redis \u4e2d\u83b7\u53d6\u7684\u9886\u5238\u7528\u6237\u8bb0\u5f55\u518d\u4fdd\u5b58\u5230 Redis\u3002<\/li>\n\n\n\n<li>\u6279\u91cf\u65b0\u589e\u65f6\u6709\u90e8\u5206\u7528\u6237\u4f18\u60e0\u5238\u5df2\u5b58\u5728\uff0c\u90a3\u4e48\u65b0\u589e\u6570\u91cf\u548c\u9884\u671f\u5c31\u4f1a\u4e0d\u4e00\u81f4\uff0c\u4e5f\u5c31\u610f\u5473\u7740\u6211\u4eec\u9700\u8981\u5c06\u8fd9\u4e9b\u5931\u8d25\u7684\u5e93\u5b58\u8fd4\u56de\u7ed9\u6570\u636e\u5e93\u548c\u7f13\u5b58\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u4e3a\u4ec0\u4e48\u6ca1\u6709\u5728\u4ee3\u7801\u903b\u8f91\u4e2d\u4f53\u73b0\u5462\uff1f\u662f\u56e0\u4e3a\u96be\u4e0d\u597d\u5199\uff0c\u6216\u8005\u9a6c\u54e5\u5077\u61d2\u4e86\u4e48\uff1f\u5176\u5b9e\u90fd\u4e0d\u662f\uff0c\u8fd9\u4e2a\u903b\u8f91\u5199\u8d77\u6765\u5f88\u7b80\u5355\uff0c\u65e0\u5916\u4e4e\u4e1a\u52a1\u5957\u4e1a\u52a1\u3002\u4e3a\u4ec0\u4e48\u4e0d\u5199\uff0c\u6211\u68b3\u7406\u4e24\u4e2a\u539f\u56e0\uff1a<\/p>\n\n\n\n<ul class=\"wp-block-list\">\n<li>\u6570\u636e\u5e93\u5b95\u673a\u53ef\u80fd\u6027\u975e\u5e38\u975e\u5e38\u4f4e\uff0c\u6211\u5de5\u4f5c\u8fd9\u4e48\u4e9b\u5e74\uff0c\u662f\u6ca1\u6709\u9047\u5230\u8fc7\u3002\u6240\u4ee5\uff0c\u4e3a\u4e86\u8fd9\u79cd\u975e\u5e38\u6781\u7aef\u7684\u6982\u5ff5\u53bb\u628a\u4e1a\u52a1\u4ee3\u7801\u53d8\u5f97\u66f4\u590d\u6742\uff0c\u6211\u8ba4\u4e3a\u662f\u5f97\u4e0d\u507f\u5931\u7684\uff0c\u4f1a\u8ba9\u5927\u5bb6\u7684\u7406\u89e3\u96be\u5ea6\u518d\u6b21\u52a0\u6df1\u3002<\/li>\n\n\n\n<li>\u4e3a\u4ec0\u4e48\u4e0d\u505a\u5e93\u5b58\u56de\u6eda\uff1f\u56e0\u4e3a\u80fd\u8fdb\u884c\u5927\u6279\u91cf\u5206\u53d1\u7684\u4f18\u60e0\u5238\u6a21\u677f\uff0c\u57fa\u672c\u4e0a\u5e93\u5b58\u53ef\u4ee5\u7406\u89e3\u4e3a\u65e0\u9650\u3002\u5982\u679c\u9700\u8981\u56de\u6eda\uff0c\u5c06\u65b0\u589e\u7684 userCouponDOList \u51cf\u53bb\u6267\u884c\u4e86 <code>batchSaveUserCouponList<\/code> \u540e\u7684 userCouponDOList \u5c31\u597d\u4e86\u3002\u4e0d\u5199\u8fd9\u4e2a\u903b\u8f91\u7684\u539f\u56e0\u540c\u4e0a\uff0c\u589e\u52a0\u4ee3\u7801\u590d\u6742\u5ea6\uff0c\u6240\u4ee5\u6ca1\u6709\u5fc5\u8981\u8fdb\u884c\u56de\u6eda\u3002<\/li>\n<\/ul>\n\n\n\n<p>\u5b8c\u7ed3\uff0c\u6492\u82b1 \ud83c\udf89<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>=_=\uff1a\u9a6c\u54e5\uff0c\u9012\u5f52\u6263\u51cf\u8fd9\u91cc\uff0c\u4ec0\u4e48\u60c5\u51b5\u4e0b\u4f1a\u4e00\u6b21\u6027\u6263\u51cf\u5931\u8d25\u8d70\u9012\u5f52\u6263\u51cf\u5462<\/p>\n\n\n\n<p>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;=_=\uff1a\u6bd4\u5982\u4f60\u6263\u51cf 5000 \u5e93\u5b58\uff0c\u4f46\u662f\u6570\u636e\u5e93\u53ea\u6709 4996 \u4e86\uff0c\u90a3\u4e48\u901a\u8fc7\u9012\u5f52\u7684\u5f62\u5f0f\u6709\u591a\u5c11\u6263\u51cf\u591a\u5c11<\/p>\n\n\n\n<p>=_=&nbsp;\u56de\u590d&nbsp;\u9a6c\u4e01\uff1a\u4f46\u662f5000\u8fd9\u4e2a\u503c\u4e0d\u662f\u5728excel\u76d1\u542c\u90a3\u91cc\u6267\u884clua\u811a\u672c\u5f97\u5230\u7684set\u5927\u5c0f\u5417\uff1f\u4e3a\u4ec0\u4e48\u7f13\u5b58\u5e93\u5b58\u591f\uff0c\u6570\u636e\u5e93\u4e0d\u591f\u5462&nbsp;<\/p>\n\n\n\n<p>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;=_=\uff1a\u7f13\u5b58\u7684\u6570\u503c\u5e76\u4e0d\u80fd\u5b8c\u5168\u53ef\u4fe1\uff0c\u53c2\u8003 AOF \u4e22\u6307\u4ee4\u548c\u96c6\u7fa4\u4e3b\u4ece\u540c\u6b65\u4e22\u6570\u636e\u573a\u666f<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u9a6c\u54e5\uff0c\u90a3\u4e2a\u6263\u51cf\u4f18\u60e0\u5377\u6a21\u7248\u5e93\u5b58\u91cf\u8fd8\u6709\u6dfb\u52a0\u5206\u53d1\u7528\u6237\u4fe1\u606f\u7684\u90a3\u4e2aLUA\u811a\u672c\u4e0d\u592a\u660e\u767d\uff1b\u91cc\u9762local userSetLength\u7684\u503c\u53ea\u4f1a\u7ef4\u6301\u57280-5000\u5417\uff1f \u662f\u56e0\u4e3a \u6bcf5000\u4e2a\u6570\u636e\u53d1\u9001\u4e00\u6b21\u6d88\u8d39\u6d88\u606f\uff0c\u7136\u540e\u6d88\u606f\u6d88\u8d39\u8005\u6d88\u8017redis\u4e2d\u7684\u5206\u53d1\u7528\u6237\u4fe1\u606f\u8bb0\u5f55\uff08\u51cf5000\uff09\u624d\u5bfc\u81f4LUA\u811a\u672c\u4e2d\u6bcf\u6b21\u83b7\u53d6userSetLength\u90fd\u662f\u57280-5000\u4ee5\u5185\u5417\uff1f\u5982\u679c\u8fd9\u4e48\u4e00\u4e2a\u60c5\u51b5\uff1a\u73b0\u5728\u626b\u63cf\u5230\u7b2c5001\u884c\u6570\u636e\uff0c\u524d\u97625000\u884c\u6570\u636e\u7684\u65f6\u5019\u5df2\u7ecf\u53d1\u9001\u6d88\u606f\u5230\u6d88\u606f\u961f\u5217\u4e86\uff0c\u4f46\u662f\u8fd8\u6ca1\u6709\u51cf\u5e93\u5b58\u91cf\uff0c\u90a3\u5c82\u4e0d\u662f\u73b0\u5728\u7684userSetLength\u8bfb\u53d6\u5230\u7684\u5c31\u662f5001\u4e86\uff1f \u90a3\u83b7\u53d6\u7b2c\u4e8c\u4e2a\u5b57\u6bb5\u7684\u503c\u5c31\u662f5001\uff0c\u5c82\u4e0d\u662f\u53c8\u4f1a\u53d1\u9001\u4e00\u6b21\u6d88\u606f\u7ed9\u6d88\u606f\u961f\u5217\uff1f \u9664\u975e\u662f\u53ef\u4ee5\u4fdd\u8bc1\u8fd9\u4e48\u4e00\u4e2a\u987a\u5e8f1.redis\u4e2d\u5206\u53d1\u7528\u6237\u4fe1\u606f\u7684key\u7684\u6570\u636e\u957f\u5ea6\u4e3a5000 2.\u53d1\u6d88\u606f\u7ed9\u6d88\u606f\u961f\u5217\u30023.\u6d88\u8d39\u8005\u6d88\u8d39\u6d88\u606f\u961f\u5217\u51cf5000. \u8fd9\u6837\u73b0\u5728\u7528\u6237\u4fe1\u606fkey\u6570\u636e\u957f\u5ea6\u4e3a0\u53c8\u80fd\u91cd\u65b0\u7edf\u8ba1\u4e86\u3002 \u6709\u70b9\u56f0\u6270\u8fd9\u91cc<\/p>\n\n\n\n<p>2024-09-07 22:45<\/p>\n\n\n\n<p>\u9a6c\u4e01&nbsp;\u56de\u590d&nbsp;Yan.\uff1a\u95ee\u9898\u63d0\u7684\u5f88\u8d5e\ud83d\udc4d main \u5206\u652f\u91cc\u6ca1\u6709\u8fd9\u4e2a\u95ee\u9898\uff0crebuild \u6211\u5ffd\u7565\u4e86\uff0c\u6211\u7406\u89e3\u52a0\u4e00\u4e2a\u53d6\u6a21\u5224\u65ad\u5c31\u597d\uff0c\u8be6\u60c5\u5982\u56fe<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"1958\" width=\"2290\" src=\"https:\/\/i-blog.csdnimg.cn\/direct\/4d8aeb8c1e6f4c2b9f578624307a1e98.png\" alt=\"\"><\/p>\n\n\n\n<p>aalwayss_&nbsp;\u56de\u590d&nbsp;Yan.\uff1a\u52a0\u5165\u67095001\u6761\u6570\u636e \u524d5000\u6761\u4fe1\u606f\u5df2\u7ecf\u88ab\u6295\u5230\u6d88\u606f\u961f\u5217\u4e86 \u6b64\u65f6\u6709\u4e09\u79cd\u60c5\u51b5\uff1a 1.\u524d5000\u6761\u6570\u636e\u5df2\u7ecf\u88abrocketMq\u5f00\u59cb\u5904\u7406\uff0c\u4f46\u662f\u6b64\u65f6\u5e93\u5b58\u603b\u91cf\u4e0d\u8db3\u4ee5\u5206\u914d\u5168\u90e8\uff08\u7f13\u5b58\u4e0d\u51c6\u7684\u60c5\u51b5\uff09 \u5047\u8bbe\u53ea\u5206\u914d\u4e00\u4e2a\u7528\u6237 \u6b64\u65f6excel\u626b\u63cf\u52305001\u884c\u6570\u636e\u5e76\u4e14set\u4e2d\u7528\u6237\u6570\u91cf\u518d\u6b21\u8fbe\u52305000\u4f1a\u518d\u6b21\u53d1\u9001rocketmq\u4f46\u4ecd\u7136\u4f1a\u9047\u5230\u6570\u636e\u5e93\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3\u60c5\u51b5\u53d1\u751f \u5982\u679c\u5206\u914d\u4e864900\u4e2a\u7528\u6237\u6b64\u65f6\u626b\u63cf\u52305001\u884c\u4e5f\u4e0d\u4f1a\u89e6\u53d1\u53d1\u9001rocketmq\u7684\u6761\u4ef6\uff0c \u53ea\u80fd\u7b49\u5f85\u7ed3\u675f\u626b\u63cf\u518d\u6b21\u53d1\u9001\u4fe1\u606f\u5230\u6d88\u606f\u961f\u5217\u4f46\u662f\u56e0\u4e3a\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3\u4e0d\u4f1a\u8fdb\u884c\u5206\u53d1\u4f18\u60e0\u5238\u800c\u8fd4\u56de \/\/2.\u524d5000\u6761\u6570\u636e\u5df2\u7ecf\u88abrocketMQ\u5f00\u59cb\u5904\u7406\uff0c\u800c\u4e14\u6570\u636e\u5145\u8db3\u5df2\u7ecf\u88abset\u4e2d\u79fb\u9664\uff0c\u6b64\u65f6\u626b\u63cf\u52305001\u884c\u6570\u636e\u65f6\uff0c\u56e0\u4e3aset\u7684\u603b\u6570\u91cf\u4e0d\u8db35000\u800c\u4e0d\u4f1a\u53d1\u9001\u6d88\u606f\u5230\u6d88\u606f\u961f\u5217\uff0c\u7b49\u5230excel\u7ed3\u675f\u626b\u63cf\u518d\u53d1\u9001\u4fe1\u606f\u5230\u6d88\u606f\u961f\u5217\uff0c\u5b8c\u6210\u5269\u4f59\u4f18\u60e0\u5238\u5206\u53d1 \/\/3.\u524d5000\u6761\u6570\u636e\u6ca1\u6709\u88abrocketMQ\u5f00\u59cb\u5904\u7406\uff0c\u6b64\u65f6\u626b\u63cf\u52305001\u884c\u65f6\uff0c\u56e0\u4e3aset\u4e2d\u7684\u6570\u636e\u91cf\u8d85\u8fc75000\uff0c\u4f1a\u518d\u6b21\u53d1\u9001\u6d88\u606f\u5230\u6d88\u606f\u961f\u5217\uff0c\u4f46\u662f\u56e0\u4e3a\u53d1\u9001\u7684\u662f\u987a\u5e8f\u6d88\u606f\u961f\u5217\uff0c\u56e0\u6b64\u5f53\u7b2c\u4e00\u6761\u6d88\u606f\u5f00\u59cb\u6d88\u8d39\u65f6\uff0c\u5982\u679c\u4f18\u60e0\u5238\u5e93\u5b58\u5145\u8db3\u5219\u5206\u53d1\uff0c\u800c\u7b2c\u4e8c\u4e2a\u5230\u6765\u7684\u6d88\u606f\u7684\u6570\u636e\u91cf\u662f5001\uff08\u53ea\u6709\u5bf95000\u53d6\u6a21\u4e3a0\u6216\u8005\u626b\u63cfexcel\u7ed3\u675f\u624d\u4f1a\u5904\u7406\uff09\u4e0d\u4f1a\u88ab\u6d88\u606f\u961f\u5217\u5904\u7406\uff0c\u7b49\u5230excel\u5b8c\u6210\u626b\u63cf\uff0c\u518d\u53d1\u9001\u6d88\u606f\u5230\u6d88\u606f\u961f\u5217\uff0c\u6700\u540e\u4e00\u6b21\u6d88\u606f\u5bf9\u4f1a\u5e26\u6570\u636e\u91cf\u662f\u591a\u5c11\uff0c\u76f4\u63a5\u53d6\u51faset\u4e2d\u7684\u6240\u6709\u6570\u636e\uff0c\u4e5f\u5c31\u662f5001\u6761\u6570\u636e\uff0c\u6700\u540e\u4e00\u6761\u6d88\u606f\u5f97\u5230\u5904\u7406 end<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;aalwayss_\uff1a\u90a3\u4e3a\u4ec0\u4e48\u6700\u540e\u4e00\u6761\u6d88\u606f\u8981\u76f4\u63a5\u628aset\u91cc\u6240\u6709\u6570\u636e\u53d6\u51fa\u6765\uff1f\u5728\u89e3\u6790\u7684\u65f6\u5019\u7b97\u597d\u6700\u540e\u8fd8\u5269\u591a\u5c11\u7136\u540e\u53d1\u5230\u6d88\u606f\u961f\u5217\uff0c\u4e0d\u5c31\u4e0d\u4f1a\u6709\u8fd9\u4e2a\u591a\u6263\u7684\u95ee\u9898\u4e86\u5417\uff1f<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u5206\u4eab\u4e00\u4e0b\u81ea\u5df1\u68b3\u7406\u7684\u903b\u8f91 \u9996\u5148\u770badmin\u540e\u7ba1\u4e2d\u7684\u6267\u884c\u6d41\u7a0b 1.CouponTemplateServiceImpl\u7c7b \u6267\u884ccreateCouponTemplate\u65b9\u6cd5\u521b\u5efa\u4e86\u4f18\u60e0\u5238\uff0c\u7136\u540e\u8ba9\u8be5\u4f18\u60e0\u5238\u4fe1\u606f\u5b58\u5165\u6570\u636e\u5e93 2.CouponTaskServiceImpl\u7c7b \u6267\u884ccreateCouponTask\u65b9\u6cd5\u521b\u5efa\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\uff0c\u7136\u540e\u8ba9\u8be5\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u5b58\u5165\u6570\u636e\u5e93 \u8fd9\u4e2a\u4efb\u52a1\u662f\u5206\u53d1\u6307\u5b9a\u7684\u4f18\u60e0\u5238\u7ed9Excel\u4e2d\u7684\u7528\u6237\uff0c\u7edf\u8ba1Excel\u4e2d\u7684\u884c\u6570\u662f\u901a\u8fc7\u7ebf\u7a0b\u6c60\u5f02\u6b65\u6267\u884c\u7684\u3002 \u5982\u679c\u5206\u53d1\u4efb\u52a1\u662f\u7acb\u5373\u53d1\u9001\uff0c\u90a3\u72b6\u6001\u5c31\u662f\u6267\u884c\u4e2d\uff0c\u5982\u679c\u4e0d\u662f\u7acb\u5373\u53d1\u9001\uff0c\u4e5f\u5c31\u662f\u5b9a\u65f6\u53d1\u9001\uff0c\u90a3\u72b6\u6001\u5c31\u662f\u5f85\u6267\u884c \u5982\u679c\u662f\u7acb\u5373\u53d1\u9001\uff0c\u5c31\u901a\u8fc7mq\u53d1\u9001\u8be5\u5206\u53d1\u4efb\u52a1id \u518d\u6765\u770bdistribution\u5206\u53d1\u7684\u6d41\u7a0b\uff0c\u6ce8\u610f\u6211\u4eec\u5206\u53d1\u7684\u65f6\u5019\uff0c\u662f\u5206\u53d1\u6307\u5b9a\u7684\u4f18\u60e0\u5238\u7ed9Excel\u4e2d\u7684\u7528\u6237 \u4e5f\u5c31\u662f\u4f18\u60e0\u5238id\u5728\u8fd9\u4e2a\u6d41\u7a0b\u4e2d\u662f\u786e\u5b9a\u7684 1.CouponTaskExecuteConsumer\u7c7b \u6267\u884conMessage\u6d88\u8d39mq\u4e2d\u7684\u6d88\u606f\uff0c\u83b7\u53d6\u5206\u53d1\u4efb\u52a1id \u7136\u540e\u6839\u636eid\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u8be2\u8be5\u4efb\u52a1 \u5224\u65ad\u4efb\u52a1\u72b6\u6001\u662f\u5426\u662f\u6267\u884c\u4e2d\uff0c\u5982\u679c\u4e0d\u662f\u6267\u884c\u4e2d\uff0c\u90a3\u5c31\u8bf4\u660e\u8fd8\u4e0d\u8be5\u6267\u884c \u5982\u679c\u4efb\u52a1\u662f\u6267\u884c\u4e2d\uff0c\u90a3\u5c31\u5224\u65ad\u4f18\u60e0\u5238\u72b6\u6001\u662f\u5426\u6b63\u786e\uff0c\u4e5f\u5c31\u662f\u770b\u4f18\u60e0\u5238\u662f\u751f\u6548\u8fd8\u662f\u7ed3\u675f \u5982\u679c\u4f18\u60e0\u5238\u751f\u6548\uff0c\u90a3\u5c31\u5f00\u59cb\u6267\u884c\u4f18\u60e0\u5238\u63a8\u9001\u4efb\u52a1\/\u5206\u53d1\u4efb\u52a1<\/p>\n\n\n\n<p>\u82b1\u5f00\u5bcc\u8d35&nbsp;\u56de\u590d&nbsp;\u6d6e\u8bed\u865a\u8f9e\uff1a2.ReadExcelDistributionListener\u7c7b \u6bcf\u83b7\u53d6\u4e00\u884cExcel\u4e2d\u7684\u6570\u636e\u5c31\u4f1a\u6267\u884cinvoke\u4e00\u6b21 \u5148\u6765\u8bf4\u6d88\u8d39\u70b9\u4f4d invoke\u4e2d\u5148\u83b7\u53d6\u6d88\u8d39\u70b9\u4f4d\uff0c\u4e5f\u5c31\u662f\u4ece\u7f13\u5b58\u4e2d\u83b7\u53d6TEMPLATE_TASK_EXECUTE_PROGRESS_KEY+couponTaskId\u7684\u503c \u5982\u679c\u6d88\u8d39\u70b9\u4f4d\u4e0d\u4e3a\u7a7a\u5e76\u4e14\u5927\u4e8e\u5f53\u524d\u5206\u53d1\u4efb\u52a1\u5904\u7406\u7684\u884c\u6570rowCount\u7684\u8bdd\uff0c\u90a3\u5c31\u8bf4\u660e\u4e4b\u524d\u5b95\u673a\u8fc7\uff0c\u4e5f\u5c31\u662f\u70b9\u4f4d\u4e4b\u524d\u7684\u90fd\u6267\u884c\u8fc7\u4e86 \u6240\u4ee5\u5982\u679c\u6ee1\u8db3\u8fd9\u79cd\u60c5\u51b5\u5c31\u76f4\u63a5return\u5e76\u4e14\u8ba9rowCount++\uff0c\u76f4\u5230rowCount\u5927\u4e8e\u6d88\u8d39\u70b9\u4f4d\uff0c\u6b64\u65f6rowCount\u5c31\u8868\u793a\u8981\u5904\u7406\u7684\u884c\u6570 \u518d\u6765\u8bf4\u7f13\u51b2\u7684\u6982\u5ff5(\u5206\u6279\u7684\u6982\u5ff5) v1\u7248\u672c\u4e2d\u6211\u4eec\u6267\u884cinvoke\u4e00\u6b21\u53ea\u4f1a\u5904\u7406\u4e00\u884c\u6570\u636e\uff0c\u56e0\u6b64\u4f1a\u9020\u6210\u78c1\u76d8IO\u7f51\u7edc\u5ef6\u8fdf\u7b49\u95ee\u9898\uff0c\u8fd9\u91cc\u6211\u4eec\u4e00\u6279\u4e00\u6279\u7684\u8fdb\u884c\u5904\u7406 \u9996\u5148\u6267\u884cstock_decrement_and_batch_save_user.lua\u811a\u672c\uff0c\u6bcf\u6b21\u6267\u884cinvoke\u65b9\u6cd5\u90fd\u4f1a\u8ba9\u8be5lua\u811a\u672c\u6267\u884c\u4e00\u6b21 \u7ed9lua\u811a\u672c\u4f20\u5165\u7684\u53c2\u6570\u4e2d\uff0ckey\u5305\u542b\u4f18\u60e0\u5238\u6a21\u677fkey\u4e5f\u5c31\u662fcouponTemplateKey\u8fd8\u6709\u7f13\u51b2\u961f\u5217\u7684key\u4e5f\u5c31\u662fbatchUserSetKey\uff0cvalue\u662f\u4e00\u4e2a\u5b58\u50a8userId\u4ee5\u53ca\u5728excel\u4e2d\u5bf9\u5e94\u884c\u6570rowNum\u7684\u8f6c\u4e3ajson\u7684map lua\u811a\u672c\u9996\u5148\u83b7\u53d6\u4f18\u60e0\u5238\u7f13\u5b58\u4e2d\u7684\u5e93\u5b58 \u5982\u679c\u5e93\u5b58\u4e3a\u7a7a\u6216\u8005\u5c0f\u4e8e0\uff0c\u90a3\u5c31\u76f4\u63a5\u8fd4\u56de(false, \u7f13\u51b2\u961f\u5217\u4e2d\u5b58\u50a8\u7684\u6570\u91cf) \u5982\u679c\u5e93\u5b58\u5927\u4e8e0\uff0c\u90a3\u5c31\u8ba9\u5e93\u5b58\u51cf1\uff0c\u7136\u540e\u8ba9value\u4e5f\u5c31\u662fuserId\u4ee5\u53carowNum\u5b58\u5165\u5230\u7f13\u51b2\u961f\u5217\u4e2d(\u5176\u5b9e\u5c31\u662f\u4e00\u4e2aset\u4e2d) \u7136\u540e\u8fd4\u56de(true, \u7f13\u51b2\u961f\u5217\u7684\u957f\u5ea6) \u5982\u679c\u6ca1\u6709\u5e93\u5b58\u7684\u8bdd\uff0c\u90a3\u4e4b\u540e\u7684\u6570\u636e\u5728\u6267\u884cinvoke\u7684\u65f6\u5019\u4e5f\u662f\u6ca1\u6709\u5e93\u5b58\u7684\uff0c\u6240\u4ee5\u6211\u4eec\u9700\u8981\u8ba9\u540e\u9762\u5206\u53d1\u5931\u8d25\u7684\u6570\u636e\u7ed9\u8bb0\u5f55\u5230\u6570\u636e\u5e93\u4e2d\uff0c\u6b64\u65f6\u4e5f\u8981\u8ba9\u5904\u7406\u5230\u7684\u884c\u6570rowCount++\uff0c\u56e0\u4e3a\u8bb0\u5f55\u5931\u8d25\u4e5f\u7b97\u5904\u7406\uff0c\u7136\u540e\u4e00\u884c\u4e00\u884c\u7684return \u5982\u679c\u8fd8\u6709\u5e93\u5b58\u7684\u8bdd\uff0c\u90a3\u5c31\u770b\u7f13\u51b2\u961f\u5217\u7684\u957f\u5ea6\u662f\u5426\u8fbe\u5230\u6211\u4eec\u8bbe\u7f6e\u7684\u6279\u6b21\u5927\u5c0fBATCH_USER_COUPON_SIZE \u5982\u679c\u6ca1\u6709\u8fbe\u5230\u6211\u4eec\u8bbe\u7f6e\u7684\u6279\u6b21\u5927\u5c0f\uff0c\u90a3\u5c31\u53ea\u66f4\u65b0\u4e00\u4e0b\u7f13\u5b58\u7684\u6d88\u8d39\u70b9\u4f4d\uff0c\u7136\u540e\u8ba9rowCount++\u5373\u53ef\uff0c\u7b80\u800c\u8a00\u4e4b\uff0c\u653e\u5230\u7f13\u51b2\u961f\u5217\u4e5f\u7b97\u5904\u7406 \u5982\u679c\u8fbe\u5230\u6211\u4eec\u8bbe\u7f6e\u7684\u6279\u6b21\u5927\u5c0f\uff0c\u90a3\u5c31\u8981\u5f00\u59cb\u5bf9\u8fd9\u4e2a\u6279\u6b21\u8fdb\u884c\u4f18\u60e0\u5238\u7684\u5206\u53d1\u4e86\uff0c\u6b64\u65f6\u5c31\u8981\u7ed9MQ\u53d1\u9001\u4e00\u6761\u6d88\u606f(\u5305\u62ec\u5206\u53d1\u4efb\u52a1id\u3001\u4f18\u60e0\u5238\u6a21\u677fid\u3001\u6279\u6b21id\u3001\u6279\u6b21\u5927\u5c0f\u3001\u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6false) excel\u5904\u7406\u5230\u6700\u540e\u624d\u4f1a\u6267\u884cdoAfterAllAnalysed\u65b9\u6cd5\uff0c\u8be5\u65b9\u6cd5\u4e5f\u662f\u7ed9MQ\u53d1\u9001\u4e00\u6761\u6d88\u606f(\u5305\u62ec\u5206\u53d1\u4efb\u52a1id\u3001\u4f18\u60e0\u5238\u6a21\u677fid\u3001\u6279\u6b21id\u3001\u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6true)\uff0c\u6b64\u65f6\u6ca1\u6709\u53d1\u9001\u6279\u6b21\u5927\u5c0f\uff0c\u800c\u662f\u540e\u9762\u7edf\u8ba1\u4e86\u4e00\u4e0b(\u8fd9\u91cc\u4e0d\u592a\u660e\u767d)<\/p>\n\n\n\n<p>\u82b1\u5f00\u5bcc\u8d35&nbsp;\u56de\u590d&nbsp;\u6d6e\u8bed\u865a\u8f9e\uff1a3.CouponExecuteDistributionConsumer\u7c7b \u8be5\u6d88\u8d39\u8005\u5c31\u662f\u7528\u6765\u6d88\u8d39ReadExcelDistributionListener\u7c7b\u53d1\u9001\u7684\u6d88\u606f\u7684\uff0c\u76f4\u63a5\u770bonMessage\u65b9\u6cd5\u5373\u53ef \u9996\u5148\u83b7\u53d6\u6d88\u606f\uff0c\u7136\u540e\u770b\u6d88\u606f\u4e2d\u5206\u53d1\u4efb\u52a1\u6807\u8bc6\u662f\u5426\u4e3afalse \u5148\u6765\u770b\u5206\u53d1\u4efb\u52a1\u6807\u8bc6\u4e3afalse\u7684\u60c5\u51b5 \u5982\u679c\u4e3afalse\u5e76\u4e14\u6d88\u606f\u4e2d\u6279\u6b21\u5927\u5c0f%BATCH_USER_COUPON_SIZE==0\uff0c\u90a3\u5c31\u8868\u793a\u540e\u7eed\u8fd8\u6709\u6570\u636e\u6ca1\u5904\u7406\uff0c\u5e76\u4e14\u5f53\u524d\u7f13\u5b58\u4e86\u4e00\u4e2a\u6279\u6b21\u7684\u7528\u6237\u6570\u636e\u4e86\uff0c\u6240\u4ee5\u76f4\u63a5\u5148\u5904\u7406\u8fd9\u4e2a\u6279\u6b21\u7684\u6570\u636e \u7136\u540e\u76f4\u63a5\u6267\u884cdecrementCouponTemplateStockAndSaveUserCouponList\u65b9\u6cd5\u8fdb\u884c\u5206\u53d1\uff0c\u8fd9\u4e2a\u65b9\u6cd5\u624d\u662f\u771f\u6b63\u5e72\u6d3b\u7684\u65b9\u6cd5\uff0c\u4f20\u5165\u7684\u53c2\u6570\u662fCouponTemplateDistributionEvent event \u8be5\u65b9\u6cd5\u9996\u5148\u6267\u884cdecrementCouponTemplateStock\u8fdb\u884c\u5e93\u5b58\u7684\u6263\u51cf\uff0c\u4f20\u5165\u7684\u53c2\u6570\u662fCouponTemplateDistributionEvent event\u4ee5\u53caevent\u4e2d\u7684\u6279\u6b21\u5927\u5c0f decrementCouponTemplateStock\u65b9\u6cd5\u901a\u8fc7\u4e50\u89c2\u9501\u673a\u5236\u6765\u8fdb\u884c\u5e93\u5b58\u7684\u6263\u51cf\uff0c\u6263\u51cf\u5931\u8d25\u5c31\u4ece\u6570\u636e\u5e93\u4e2d\u67e5\u8be2\u6700\u65b0\u7684\u5e93\u5b58\uff0c\u7136\u540e\u9012\u5f52\u8c03\u7528\u81ea\u8eab\u7528\u5f53\u524d\u5e93\u5b58\u6570\u91cf\u91cd\u65b0\u6263\u51cf\uff0c\u9012\u5f52\u7684\u76ee\u7684\u662f\u4e3a\u4e86\u7f29\u5c0f\u6263\u51cf\u6570\u91cf\uff0c\u76f4\u5230\u5e93\u5b58\u6263\u51cf\u6216\u5e93\u5b58\u4e3a0 decrementCouponTemplateStock\u65b9\u6cd5\u6700\u7ec8\u8fd4\u56de\u7684\u662f\u5b9e\u9645\u6263\u51cf\u7684\u6570\u91cf\uff0c\u7136\u540e\u8d4b\u503c\u7ed9couponTemplateStock \u5982\u679ccouponTemplateStock\u5c0f\u4e8e\u7b49\u4e8e0\uff0c\u8868\u793a\u5e93\u5b58\u4e0d\u591f\u4e86\uff0c\u90a3\u5c31decrementCouponTemplateStockAndSaveUserCouponList\u65b9\u6cd5\u76f4\u63a5return\u9000\u51fa\u4e0d\u5e72\u6d3b \u5982\u679ccouponTemplateStock\u5927\u4e8e0\uff0c\u8868\u793a\u6263\u51cf\u6210\u529f\u4e86couponTemplateStock\u4e2a(\u53ef\u80fd\u5e76\u4e0d\u662f\u90fd\u6263\u51cf\u6210\u529f\u4e86) \u90a3\u5c31\u4ece\u7f13\u51b2\u961f\u5217\u91cc\u9762pop\u51facouponTemplateStock\u4e2a\u6570\u636e\uff0c\u8d4b\u503c\u7ed9List&lt;String&gt; batchUserMaps\uff0c\u56e0\u4e3acouponTemplateStock\u4e2a\u80af\u5b9a\u90fd\u662f\u6263\u51cf\u6210\u529f\u7684\uff0c\u6240\u4ee5\u53ef\u4ee5\u653e\u5fc3\u5904\u7406 \u5904\u7406\u7684\u65b9\u5f0f\u5c31\u662f\uff0c\u904d\u5386\u8fd9\u4e2a\u96c6\u5408\uff0c\u7136\u540e\u8f6c\u56demap(\u56e0\u4e3a\u5b58\u5165\u7684\u65f6\u5019\u5c31\u662f\u5b58\u50a8userId\u4ee5\u53ca\u5728excel\u4e2d\u5bf9\u5e94\u884c\u6570rowNum\u7684\u8f6c\u4e3ajson\u7684map) \u7136\u540e\u8ba9\u5bf9\u5e94\u7684\u884c\u53f7\u3001\u7528\u6237id\u3001\u4f18\u60e0\u5238id\u3001\u9886\u53d6\u6570\u91cf\u4e5f\u5c31\u662f1\u7b49\u4fe1\u606f\u6784\u5efa\u4e3aUserCouponDO\u7c7b\u7136\u540e\u5b58\u5165List&lt;UserCouponDO&gt; userCouponDOList\u96c6\u5408\u4e2d\uff0c\u6700\u540e\u6279\u91cf\u63d2\u5165\u7528\u6237\u4f18\u60e0\u5238\u8868\u4e2d(\u4e5f\u5c31\u662f\u6267\u884cbatchSaveUserCouponList\u65b9\u6cd5)\uff0c\u8fd9\u5c31\u8868\u793a\u9886\u53d6\u6210\u529f\u4e86<\/p>\n\n\n\n<p>\u82b1\u5f00\u5bcc\u8d35&nbsp;\u56de\u590d&nbsp;\u6d6e\u8bed\u865a\u8f9e\uff1abatchSaveUserCouponList\u65b9\u6cd5\u4e5f\u5f88\u6709\u8003\u7a76\uff0c\u76f4\u63a5\u6267\u884cuserCouponMapper.insert(userCouponDOList, userCouponDOList.size())\u65b9\u6cd5\u8fdb\u884c\u6279\u91cf\u63d2\u5165 \u5982\u679c\u6279\u91cf\u63d2\u5165\u6709\u9519\uff0c\u90a3\u5c31\u901a\u8fc7\u4e00\u6761\u4e00\u6761\u63d2\u5165\u6267\u884c\uff0c\u6240\u4ee5\u904d\u5386userCouponDOList\u6765\u8fdb\u884c\u63d2\u5165\u3002\u5982\u679c\u5355\u6311\u63d2\u5165\u4e5f\u5931\u8d25\u7684\u8bdd\uff0c\u90a3\u5c31\u6267\u884chasUserReceivedCoupon\u770b\u7528\u6237\u662f\u5426\u9886\u53d6\u8fc7\u4f18\u60e0\u5238 \u5982\u679c\u5df2\u7ecf\u9886\u53d6\u8fc7\u4e86\uff0c\u90a3\u5c31\u8ba9\u8be5\u6761\u8bb0\u5f55\u7684\u884c\u53f7\u4ee5\u53ca\u5bf9\u5e94\u9519\u8bef\u539f\u56e0\u4e5f\u5c31\u662f\u7528\u6237\u5df2\u9886\u53d6\u8be5\u4f18\u60e0\u5238\u52a0\u5165\u5230List&lt;CouponTaskFailDO&gt; couponTaskFailDOList\u4e2d\uff0c\u6700\u540e\u6279\u91cf\u63d2\u5165\u6570\u636e\u5e93 \u56e0\u4e3a\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u8fc7\u4e86\uff0c\u90a3\u5c31\u6dfb\u52a0\u5230List&lt;UserCouponDO&gt; toRemove\u8868\u793a\u5df2\u7ecf\u5904\u7406\u8fc7\u4e86\uff0c\u4e4b\u540e\u8981\u5728List&lt;UserCouponDO&gt; userCouponDOList\u96c6\u5408\u4e2d\u79fb\u9664 \u6240\u4ee5\u6267\u884c\u5b8cbatchSaveUserCouponList\u65b9\u6cd5\u4e4b\u540e\uff0c\u6b64\u65f6userCouponDOList\u5df2\u7ecf\u5254\u9664\u4e86\u91cd\u590d\u7684\u6216\u8005\u5df2\u7ecf\u6dfb\u52a0\u7684\u8bb0\u5f55\uff0c\u7136\u540e\u5c06\u8fd9\u4e9b\u4f18\u60e0\u5238\u9886\u53d6\u8bb0\u5f55\u6dfb\u52a0\u5230\u7f13\u5b58\u4e2d\uff0c \u56e0\u4e3a\u6570\u636e\u5e93\u5c42\u9762\u52a0\u5165\u7528\u6237\u4f18\u60e0\u5238\u8bb0\u5f55\u5df2\u7ecf\u5728\u4e0a\u9762\u505a\u4e86\uff0c\u8fd9\u91cc\u5c31\u662f\u6267\u884cbatch_user_coupon_list.lua\u811a\u672c \u518d\u6765\u770b\u5206\u53d1\u4efb\u52a1\u6807\u8bc6\u4e3atrue\u7684\u60c5\u51b5 \u5206\u53d1\u4efb\u52a1\u7ed3\u675f\u6807\u8bc6\u4e3atrue\u6807\u8bc6\u6ca1\u6709Excel\u8bb0\u5f55\u4e86\uff0c\u8bf4\u660eExcel\u6570\u636e\u8bfb\u53d6\u7ed3\u675f\uff0c\u6574\u4e2a\u5206\u53d1\u6d41\u7a0b\u8d70\u5230\u4e86\u6700\u540e\u9636\u6bb5\uff0c\u9700\u8981\u5904\u7406\u5269\u4e0b\u7684\u7f13\u51b2\u961f\u5217\u4e2d\u6ca1\u6709\u53d1\u51fa\u53bb\u7684\u7528\u6237\u6570\u636e \u9996\u5148\u83b7\u53d6\u8fd9\u4e2a\u6279\u6b21\u7684\u6570\u636e\u6570\u91cf\uff0c\u7136\u540e\u8d4b\u503c\u7ed9event \u7136\u540e\u6267\u884cdecrementCouponTemplateStockAndSaveUserCouponList\u65b9\u6cd5\u8fdb\u884c\u5206\u53d1\uff0c\u8fc7\u7a0b\u4e0d\u5728\u8d58\u8ff0\uff0c\u548c\u4e0a\u9762\u4e00\u6837\uff0c\u6700\u540e\u5c31\u662f\u8ba9\u5206\u53d1\u4f18\u60e0\u5238\u6210\u529f\u7684\u6570\u636e\u7ed9\u5b58\u5165\u7528\u6237\u4f18\u60e0\u5238\u8868\u4e2d \u7136\u540e\u518d\u4ece\u7f13\u51b2\u961f\u5217\u91cc\u9762pop\u51faInteger.MAX_VALUE\u4e2a\u6570\u636e\uff0c\u4e5f\u5c31\u662f\u5168\u90e8pop\u51fa\u53bb\uff0c\u8fd9\u4e9b\u5c31\u662f\u4e0d\u80fd\u591f\u6b63\u786e\u5904\u7406\u7684\uff0c\u4e5f\u5c31\u662f\u5e93\u5b58\u4e0d\u8db3\u6216\u8005\u7528\u6237\u5df2\u7ecf\u9886\u53d6\u4f18\u60e0\u5238\u7684\u539f\u56e0\uff0c \u800c\u6267\u884cbatchSaveUserCouponList\u7684\u65f6\u5019\u5df2\u7ecf\u8ba9\u7528\u6237\u9886\u53d6\u4f18\u60e0\u5238\u7684\u60c5\u51b5\u7ed9\u8003\u8651\u8fdb\u53bb\u4e86\uff0c\u6240\u4ee5\u8fd9\u91cc\u7684\u539f\u56e0\u5c31\u662f\u5e93\u5b58\u4e0d\u8db3\uff0c\u7136\u540e\u8ba9\u9886\u53d6\u5931\u8d25\u7684\u8bb0\u5f55\u7ed9\u6279\u91cf\u5b58\u5165\u6570\u636e\u5e93\u4e2d \u6267\u884c\u5230\u6700\u540e\uff0c\u8ba9\u8be5\u5206\u53d1\u4efb\u52a1\u72b6\u6001\u6539\u4e3a\u5df2\u5b8c\u6210\uff0c\u5e76\u4e14\u66f4\u65b0\u76f8\u5e94\u5b57\u6bb5\u5373\u53ef<\/p>\n\n\n\n<p>\u3002&nbsp;\u56de\u590d&nbsp;\u82b1\u5f00\u5bcc\u8d35\uff1a\u2018\u5982\u679ccouponTemplateStock\u5c0f\u4e8e\u7b49\u4e8e0\uff0c\u8868\u793a\u5e93\u5b58\u4e0d\u591f\u4e86\uff0c\u90a3\u5c31decrementCouponTemplateStockAndSaveUserCouponList\u65b9\u6cd5\u76f4\u63a5return\u9000\u51fa\u4e0d\u5e72\u6d3b\u2019\u90a3\u8fd9\u4e9b\u4e0d\u5e94\u8be5\u4e5f\u63d2\u5165\u5230\u7528\u6237\u5206\u53d1\u5931\u8d25\u8868\u4e2d\u5417\uff1f\u8fd9\u91cc\u597d\u50cf\u4ec0\u4e48\u4e5f\u6ca1\u5904\u7406<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u7591\u95ee\uff1a\u6587\u4e2d\u8bf4\u4e86\u4f7f\u7528\u987a\u5e8f\u6d88\u606f\uff0c\u4f46\u662f\u4ee3\u7801\u4e2d\u6ca1\u6709\u770b\u5230\u54c8\u5e0c\u9009\u62e9\u961f\u5217\u7684\u4ee3\u7801\u3002\u987a\u5e8f\u6d88\u606f\u662f\u600e\u4e48\u7528\u7684\uff1f<\/p>\n\n\n\n<p>o3o&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u53d1\u9001\u6d88\u606f\u7684\u65f6\u5019\uff0c\u91cd\u5199MessageSelector\u7c7b\u5373\u53ef \u5c31\u53ef\u4ee5\u53d1\u5230\u6307\u5b9a\u961f\u5217<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;o3o\uff1a\u633a\u9ebb\u70e6\uff0c\u6211\u76f4\u63a5\u7528springboot\u7684api\uff0c\u8fd9\u4e2a\u91cc\u9762\u7684\u9009\u62e9\u961f\u5217\u7b97\u6cd5\u9ed8\u8ba4\u5c31\u662f\u53d6\u6a21<\/p>\n\n\n\n<p>o3o&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u5f88\u7b80\u5355\u5440 \u5c31\u51e0\u884c\u641e\u5b9a<\/p>\n\n\n\n<p>O\u6ce1\u679c\u5976&nbsp;\u56de\u590d&nbsp;o3o\uff1a\u8bf7\u95ee\u4ee3\u7801\u91cc\u987a\u5e8f\u6d88\u606f\u548b\u505a\u7684\uff0c\u6ca1\u770b\u5565\u5904\u7406\u5440\uff0c\u662f\u5355\u961f\u5217\u5417<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;o3o\uff1a\u5176\u5b9e\uff0cspringboot\u96c6\u6210\u7684mq\uff0c\u91cc\u9762\u961f\u5217\u9009\u62e9\u7684\u65b9\u6cd5\u9ed8\u8ba4\u5c31\u662f\u54c8\u5e0c\u53d6\u6a21\uff0c\u76f4\u63a5\u7528\u5c31\u884c\u4e86<\/p>\n\n\n\n<p>\u9e45\u9e45\u9e45&nbsp;\u56de\u590d&nbsp;O\u6ce1\u679c\u5976\uff1a\u76f4\u63a5\u7528\u5e26orderly\u7684\u90a3\u4e2a\u65b9\u6cd5\uff0c\u6307\u5b9a\u4e2a\u5206\u7247\u952e\uff0c\u7136\u540e\u6d88\u8d39\u8005\u914d\u987a\u5e8f\u6d88\u8d39\u6a21\u5f0f<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u6211\u6709\u4e00\u4e2a\u7591\u95ee\uff0c\u770b\u4ee3\u7801\u903b\u8f91\u5e94\u8be5\u662f\u6bcf\u8bfb\u53d6\u4e00\u884cexcel\u5c31\u6267\u884c\u4e00\u6b21lua\uff0c\u5b9e\u65f6\u6263\u51cf\u5e93\u5b58\uff0c\u4f46\u662f\u53ea\u6709\u9886\u5238\u8bb0\u5f55\u957f\u5ea6\u8fbe\u52305000\u624d\u4e00\u6b21\u6027\u6279\u91cf\u5199\u5165\u6570\u636e\u5e93\uff0c\u8fd9\u6837\u770bmysql\u7684\u66f4\u65b0\u76f8\u5f53\u4e8e\u5ef6\u8fdf\u7684\uff0c\u90a3\u5982\u679c\u4e2d\u9014\u5b95\u673a\u4e86\uff0credis\u5df2\u7ecf\u6263\u51cf\u4e86\u5e93\u5b58\u4f46mysql\u5e76\u672a\u6267\u884c\uff0c\u8fd9\u6837\u662f\u4e0d\u662f\u4f1a\u5bfc\u81f4\u5e93\u5b58\u6570\u636e\u548c\u6570\u636e\u5e93\u4e2d\u6570\u636e\u4e0d\u540c\u6b65\u5440<\/p>\n\n\n\n<p>\u5b64\u8fdc&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u4e2d\u9014\u5b95\u673a\u7684\u8bdd\uff0c\u4f1a\u5bfc\u81f4\u6d88\u8d39\u8005\u672a\u80fd\u6b63\u5e38\u6d88\u8d39\u6d88\u606f\uff0c\u4f46\u540e\u9762\u8fd8\u4f1a\u518d\u6b21\u6d88\u8d39\u8fd9\u6761\u4fe1\u606f\u7684<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u54e5\u4eec\u513f\uff0c\u4f60\u4e0d\u89c9\u5f97\u8fd9\u4e2a\u5904\u7406\u903b\u8f91\u6709\u95ee\u9898\u5417\uff1f\u8bfb\u6587\u4ef6\u662f\u5355\u7ebf\u7a0b\u7684\uff0c\u540e\u9762\u6279\u5904\u7406\u7684\u610f\u4e49\u4e0d\u662f\u7279\u522b\u5927\u554a\uff0c\u800c\u4e14easyexcel\u7684\u91cd\u5199\u65b9\u6cd5\u7684\u7b2c\u4e8c\u4e2a\u53c2\u6570\u5c31\u662f\u5206\u6790\u5bb9\u5668\uff0c\u53ef\u4ee5\u83b7\u53d6\u5f53\u524d\u8fdb\u5ea6\u6761\u7684\u884c\u53f7\u548csheet\u53f7<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u538b\u6839\u5c31\u6ca1\u5fc5\u8981\u5404\u79cd++\u8ba1\u7b97<\/p>\n\n\n\n<p>Fortitude&nbsp;\u56de\u590d&nbsp;\u5b64\u8fdc\uff1a\u90a3\u8fd9\u91cc\u662f\u4e0d\u4f1a\u6709\u4e0d\u4e00\u81f4\u98ce\u9669\u5417<\/p>\n\n\n\n<p>Fortitude&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u4f60\u7684\u610f\u601d\u662fredis\u6263\u51cf\u5e93\u5b58\u7684\u65f6\u5019\u4e5f\u4e0d\u4e00\u4e2a\u4e00\u4e2a\u51cf\u5417\uff1f<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u6709\u5fc5\u8981\u5417\uff1f<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u76f4\u63a5\u628alist\u957f\u5ea6\u6263\u51cf\u5c31\u597d\u4e86<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u800c\u4e14\u8bfbeasyExcel\u4e5f\u5e94\u8be5\u7528\u7ebf\u7a0b\u6c60\u6279\u5904\u7406\uff0c\u54ea\u6709\u5355\u7ebf\u7a0b\u8bfb\u6587\u4ef6\uff0c\u7136\u540e\u4e00\u987a\u6487\u90fd\u662f\u987a\u5e8f\u6d88\u8d39\u7684\uff1f\u5355\u7ebf\u7a0b\u5904\u7406\u767e\u4e07\u6570\u636e\uff1f<\/p>\n\n\n\n<p>Fortitude&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u5bf9\u3002\u3002\u3002\u8fd9\u4e2a\u7591\u95ee\u6211\u4e5f\u6709\u3002\u3002\u3002<\/p>\n\n\n\n<p>Fortitude&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u611f\u89c9\u5355\u7ebf\u7a0b\u4e00\u884c\u4e00\u884c\u8bfb\u6709\u70b9\u602a<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u4f60\u653e\u5fc3\uff0c\u666e\u901awin\u7cfb\u7edf\u768413900\u53ca\u4ee5\u4e0b\u7684CPU\u538b\u6839\u8dd1\u4e0d\u51fa\u9a6c\u54e5\u7684\u6027\u80fd\uff0c\u6211\u8ddf\u5e08\u5144\u4f18\u5316\u4e86\u597d\u4e45\u624d\u628a\u901f\u5ea6\u63d0\u5347\u51e0\u5341\u500d<\/p>\n\n\n\n<p>Fortitude&nbsp;\u56de\u590d&nbsp;int main()\uff1a\u6240\u4ee5\u4f60\u4eec\u6d4b\u51fa\u6765\u7684\u901f\u5ea6\u662f\u591a\u5c11\u54c7<\/p>\n\n\n\n<p>int main()&nbsp;\u56de\u590d&nbsp;Fortitude\uff1a\u516d\u4e03\u5206\u949f\u5427\uff0c\u5c31\u80fd\u5206\u53d1\u5b8c\u767e\u4e07\u6570\u636e<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u770b\u4e86\u4e24\u4e09\u904d\u7b97\u662f\u770b\u61c2\u4e86\uff0c\u6574\u4e2a\u6267\u884c\u6d41\u7a0b\u5927\u6982\u662f\u8fd9\u6837 : (1)-&gt;\uff08\u4e0a\u4e00\u7ae0\u7684\u5185\u5bb9\uff09\u540e\u7ba1\u670d\u52a1\u521b\u5efa\u5206\u53d1\u4efb\u52a1\u8868\u63d2\u5165\u5230\u6570\u636e\u5e93 -&gt; \u7edf\u8ba1excel\u884c\u6570 -&gt; \u5c06\u884c\u6570\u63d2\u5165\u4f18\u60e0\u5238\u5206\u53d1\u4efb\u52a1\u8868 -&gt;\u53d1\u9001\u5b9a\u65f6\u6d88\u606f\u6216\u5ef6\u65f6\u6d88\u606f\u6267\u884c\u7ed9\u5206\u53d1\u5f15\u64ce\u670d\u52a1\u6267\u884c\u5206\u53d1\u4efb\u52a1\u3002 (2) -&gt; \u6821\u9a8c\u6d88\u8d39\u8005\u62ff\u5230\u6d88\u606f\u5e76\u8fdb\u884c\u6821\u9a8c -&gt; \u4f7f\u7528easyexal\u8bfb\u53d6 -&gt; Easyexcel\u76d1\u542c\u7c7b \u8bb0\u5f55\u6267\u884c\u70b9\u4f4d \u3001\u6267\u884c\u7f13\u5b58\u6263\u51cf\uff0c\u8bb0\u5f55\u6279\u91cf\u6570\u636e\u5230\u7f13\u5b58 -&gt; \u6279\u91cf\u8fbe\u52305000\uff0c\u53d1\u9001\u6d88\u606f\u7ed9\u5206\u53d1\u6d88\u8d39\u8005 -&gt; \u5206\u53d1\u6d88\u8d39\u8005\u62ff\u5230\u6d88\u606f(\u987a\u5e8f\u6267\u884c) -&gt; \u6279\u91cf\u4fdd\u5b58\u5230\u6570\u636e\u5e93 -&gt;\u8981\u662f\u5e93\u5b58\u4e0d\u8db3\uff0c\u6279\u91cf\u8bb0\u5f55\u5230\u53d1\u653e\u5931\u8d25\u8868 -&gt; \u66f4\u65b0\u5206\u53d1\u4efb\u52a1\u8868\u7684\u5b8c\u6210\u65f6\u95f4<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"2205\" width=\"3110\" src=\"https:\/\/i-blog.csdnimg.cn\/direct\/cedd2197ad704b98b353e6d1b499060d.png\" alt=\"\"><\/p>\n\n\n\n<p>\u5927\u4f53\u7684\u601d\u8def\u5c31\u662f\u5c06\u7528\u6237\u662f\u5426\u5df2\u9886\u8fc7\u5238\u7684\u5224\u65ad\u63d0\u524d\uff0c\u4e0d\u8981\u5728\u6279\u91cf\u63d2\u5165\u5f02\u5e38\u65f6\u518d\u505a\u5904\u7406\uff0c\u7531\u4e8e\u6211\u7684\u4ee3\u7801\u8ddf\u9a6c\u54e5\u7684\u4e0d\u592a\u4e00\u6837\uff0c\u6539\u4e86\u4e00\u4e9b\u4e1c\u897f\uff0c\u5927\u5bb6\u53ef\u4ee5\u5bf9\u7167\u7740\u770b\u4e00\u4e0b\u3002<\/p>\n\n\n\n<p>\u5148\u4e0a\u6d4b\u8bd5\u7ed3\u679c\uff0c\u5206\u53d1\u7ed91000\u4e2a\u7528\u6237\uff0c\u5176\u4e2d\u670920\u4e2a\u7528\u6237\u5df2\u9886\u8fc7\u8be5\u5238\uff0c\u9884\u671f\u7ed3\u679c\u5206\u53d1\u6210\u529f980\u5f20\uff0c<strong>coupon_distribution_task_fail<\/strong>\u8868\u4e2d\u65b0\u589e20\u6761\u53d1\u653e\u5931\u8d25\u7684\u8bb0\u5f55\uff0c\u7f13\u5b58\u548c\u6570\u636e\u5e93\u4e2d\u4f18\u60e0\u5238\u5e93\u5b58\u5269\u4f5920\u5f20\u3002<\/p>\n\n\n\n<p>\u9996\u5148\u521b\u5efa\u4e00\u5f20\u4f18\u60e0\u5238\uff0c\u7f13\u5b58\u548cdb\u4e2d\u7684\u5e93\u5b58\u90fd\u4e3a20\uff1b<\/p>\n\n\n\n<p>\u4ece1000\u4e2a\u7528\u6237\u4e2d\u62bd20\u4e2a\u7528\u6237\u5148\u6267\u884c\u4e00\u4e0b\u63a8\u9001\u4efb\u52a1\uff0c\u5206\u53d1\u6210\u529f\u540e\u4f1a\u5411\u7f13\u5b58<strong>user-gained-coupon<\/strong>\u5199\u5165\u7528\u6237\u9886\u5238\u8bb0\u5f55\uff1b<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"345\" width=\"1118\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/74d7f67d9356144b7e28d32d66770885.png\" alt=\"image.png\"><\/p>\n\n\n\n<p>\u4e3a\u8fd9\u5f20\u5238\u65b0\u589e1000\u5f20\u5e93\u5b58\uff0c\u518d\u6b21\u6267\u884c\u63a8\u9001\u4efb\u52a1\uff0c\u7f13\u5b58\u548cdb\u4e2d\u7684\u5e93\u5b58\u90fd\u53d8\u6210\u4e8620\uff0c<strong>coupon_distribution_task_fail<\/strong>\u8868\u4e2d\u591a\u4e8620\u6761\u8bb0\u5f55\uff0c\u6b63\u662f\u4e4b\u524d\u9886\u8fc7\u5238\u7684\u7528\u6237\uff1b<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"701\" width=\"1122\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/c8017fc444ac58574f22de85ca6b7cfd.png\" alt=\"image.png\"><\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"74\" width=\"1196\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/16eafef857cc91c4ff5d4015927123b0.png\" alt=\"image.png\"><\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"591\" width=\"1412\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/188d520d7d21974e4b97fc763896a2f6.png\" alt=\"image.png\"><\/p>\n\n\n\n<p>\u6267\u884c\u65f6\u95f437ms\uff08\u53ef\u80fd\u662f\u56e0\u4e3a\u4e4b\u524d\u8bf7\u6c42\u8fc7\u4e00\u6b21\u7684\u539f\u56e0\uff0c\u7b2c\u4e00\u6b21\u6267\u884c\u662f\u5728700ms\uff09\uff1b<\/p>\n\n\n\n<p><img loading=\"lazy\" decoding=\"async\" height=\"800\" width=\"1280\" src=\"https:\/\/i-blog.csdnimg.cn\/img_convert\/1671c7bff016a8227f1bb4ca41c8a5cc.png\" alt=\"image.png\"><\/p>\n\n\n\n<p>\u4ee3\u7801\u4e3b\u8981\u5206\u4e3a<strong>ExcelDistributionListener\uff08\u8bfb\u53d6excel\u6bcf\u4e00\u884c\u6570\u636e\uff0c\u505a\u4e00\u4e9b\u524d\u7f6e\u5de5\u4f5c\uff09<\/strong>\u548c<strong>CouponDispenseConsumer\uff08\u771f\u6b63\u6267\u884c\u5206\u53d1\uff09<\/strong><\/p>\n\n\n\n<p><code>@RequiredArgsConstructor public class ExcelDistributionListener extends AnalysisEventListener&lt;CouponDistributionUserExcelBO&gt; { private final CouponTemplateDO couponTemplateDO; private final RedisProxy redisProxy; private final CouponDistributionFailMapper couponDistributionFailMapper; private final CouponDistributionTaskDO taskDO; private final CouponDispenseProducer couponDispenseProducer; \/** * \u8df3\u8fc7\u6807\u9898\uff0c\u4ece\u7b2c2\u884c\u5f00\u59cb\u8bfbexcel *\/ private long rowRecord = 2; \/** * \u5206\u6279\u6b21\u6267\u884c\u7684\u5927\u5c0f *\/ private static final int BATCH_SIZE = 200; \/** * \u5206\u53d1\u5931\u8d25\u8bb0\u5f55\u7684\u6682\u5b58\u533a *\/ private final List&lt;CouponDistributionTaskFailDO&gt; failRecordCacheList = new ArrayList&lt;&gt;(BATCH_SIZE); @Override public void invoke(CouponDistributionUserExcelBO excelData, AnalysisContext context) { String couponTemplateKey = StrUtil.join(\":\", \"coupon-template\", couponTemplateDO.getId()); String processKey = StrUtil.join(\":\", \"coupon-distribution-process\", taskDO.getId()); if (redisProxy.exist(couponTemplateKey)) { \/\/ \u83b7\u53d6\u5f53\u524d\u6267\u884c\u8fdb\u5ea6\uff0c\u4f18\u5316\u56e0\u7cfb\u7edf\u91cd\u542f\u6216\u5b95\u673a\u800c\u91cd\u590d\u6267\u884c\u5206\u53d1 String point = redisProxy.get(processKey, String.class); if (point != null &amp;&amp; Integer.parseInt(point) &gt;= rowRecord) { rowRecord++; return; } \/\/ \u83b7\u53d6lua\u811a\u672c\u5bf9\u8c61\u5e76\u4fdd\u5b58\u5230\u5355\u4f8b\u5bb9\u5668\uff0c\u4f18\u5316\u6bcf\u6b21\u90fd\u9700\u8981\u91cd\u65b0\u521b\u5efa\u811a\u672c\u5bf9\u8c61 DefaultRedisScript&lt;Long&gt; luaScript = Singleton.get(AdminConstantPool.COUPON_DISTRIBUTION_TASK_LUA_SCRIPT, () -&gt; { DefaultRedisScript&lt;Long&gt; script = new DefaultRedisScript&lt;&gt;(); script.setScriptSource(new ResourceScriptSource(new ClassPathResource(AdminConstantPool.COUPON_DISTRIBUTION_TASK_LUA_SCRIPT))); script.setResultType(Long.class); return script; }); String batchUserReceiveCouponKey = StrUtil.join(\":\", \"user-coupon-collection\", taskDO.getId()); String userId = excelData.getUserId(); Map&lt;Object, Object&gt; userRowNumMap = Collections.emptyMap(); \/\/ \u8ba1\u7b97\u7528\u6237\u9886\u5238\u96c6\u5408\u7f13\u5b58\u5206\u7247\u952e\u5e76\u5224\u65ad\u5f53\u524d\u7528\u6237\u662f\u5426\u5df2\u7ecf\u9886\u8fc7\u8be5\u4f18\u60e0\u5238 int mod = (Long.valueOf(userId).hashCode() &amp; Integer.MAX_VALUE) % 5; String key = StrUtil.join(\":\", \"user-gained-coupon\", mod); String hashKey = StrUtil.join(\"-\", userId, couponTemplateDO.getId()); \/\/ \u5982\u679c\u5df2\u9886\u53d6\u8fc7\u5219\u5c06\u5176\u4fdd\u5b58\u4e3a\u53d1\u5238\u5931\u8d25\u7684\u8bb0\u5f55 if (redisProxy.getTemplateInstance().opsForHash().hasKey(key, hashKey)) { Map&lt;Object, Object&gt; contentMap = MapUtil.builder() .put(\"userId\", userId) .put(\"cause\", \"\u7528\u6237\u5df2\u9886\u53d6\u8fc7\u8be5\u4f18\u60e0\u5238\") .build(); CouponDistributionTaskFailDO taskFailDO = CouponDistributionTaskFailDO.builder() .batchId(taskDO.getBatchId()) .content(JSONUtil.toJsonStr(contentMap)) .build(); failRecordCacheList.add(taskFailDO); } else { userRowNumMap = Collections.singletonMap(\"userId\", userId); } if (userRowNumMap.isEmpty()) { \/\/ \u68c0\u67e5\u4e00\u4e0b\u5931\u8d25\u8bb0\u5f55\u6682\u5b58\u533a\u7684\u5927\u5c0f\uff0c\u53ca\u65f6\u8fdb\u884c\u6279\u5904\u7406\uff0c\u9632\u6b62\u5728\u6781\u7aef\u60c5\u51b5\u4e0b\u5806\u79ef\u5927\u91cf\u5931\u8d25\u8bb0\u5f55\u5bfc\u81f4ArrayList\u6269\u5bb9 if (failRecordCacheList.size() % BATCH_SIZE == 0) { couponDistributionFailMapper.insert(failRecordCacheList); failRecordCacheList.clear(); } return; } \/\/ \u6263\u51cf\u7f13\u5b58\u4e2d\u7684\u4f18\u60e0\u5238\u5e93\u5b58\u4ee5\u53ca\u65b0\u589e\u8bb0\u5f55\u5230\u7528\u6237\u9886\u5238\u96c6\u5408 Long combineInt = redisProxy.getTemplateInstance().execute(luaScript, Arrays.asList(couponTemplateKey, batchUserReceiveCouponKey), JSONUtil.toJsonStr(userRowNumMap)); assert combineInt != null; boolean hasInventory = BinarySplitUtil.extractFirstField(combineInt); if (!hasInventory) { failRecord(processKey, userId, \"\u7f13\u5b58\u4e2d\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3\"); return; } int setSize = BinarySplitUtil.extractSecondField(combineInt); if (setSize &lt; BATCH_SIZE) { syncProcess(processKey); return; } \/\/ \u901a\u77e5\u53e6\u4e00\u4e2a\u6d88\u8d39\u8005\u6267\u884c\u5206\u53d1\uff0c\u907f\u514d\u5355\u6761\u6d88\u606f\u6d88\u8d39\u65f6\u95f4\u8fc7\u957f\uff0c\u8fd9\u91cc\u53d6\u6a21\u53ef\u4ee5\u4fdd\u8bc1\u6bcf5000\u4e3a\u4e00\u4e2a\u6279\u6b21\u7684\u53d1\u6d88\u606f\uff0c\u6700\u540e\u4e0d\u8db35000\u7684\u4ea4\u7ed9\u540e\u7f6e\u903b\u8f91\u6765\u5904\u7406 if (setSize % BATCH_SIZE == 0) { CouponDispenseMessage messageEvent = CouponDispenseMessage.builder() .taskId(taskDO.getId()) .shopId(couponTemplateDO.getShopId()) .couponTemplateId(couponTemplateDO.getId()) .taskBatchId(taskDO.getBatchId()) .batchSetSize(BATCH_SIZE) .endFlag(false) .build(); couponDispenseProducer.sendOrderlyMessage(messageEvent); } syncProcess(processKey); } else { failRecord(processKey, null,\"\u4f18\u60e0\u5238\u7f13\u5b58\u5931\u6548\"); } } @Override public void doAfterAllAnalysed(AnalysisContext context) { CouponDispenseMessage messageEvent = CouponDispenseMessage.builder() .taskId(taskDO.getId()) .shopId(couponTemplateDO.getShopId()) .couponTemplateId(couponTemplateDO.getId()) .taskBatchId(taskDO.getBatchId()) .endFlag(true) .build(); couponDispenseProducer.sendOrderlyMessage(messageEvent); if (!failRecordCacheList.isEmpty()) { couponDistributionFailMapper.insert(failRecordCacheList); failRecordCacheList.clear(); } } \/** * \u4fdd\u5b58\u5206\u53d1\u5931\u8d25\u7684\u8bb0\u5f55 * @describe \u6279\u91cf\u6267\u884c\u6700\u4fdd\u9669\u7684\u65b9\u6cd5\u5e94\u8be5\u662f\u53d1mq\uff0c\u4f46\u6b63\u5e38\u6267\u884c\u51e0\u4e4e\u4e0d\u53ef\u80fd\u4f1a\u5206\u53d1\u5931\u8d25 *\/ private void failRecord(String processKey, String userId, String cause) { Map&lt;Object, Object&gt; contentMap = MapUtil.builder() .put(\"userId\", userId) .put(\"cause\", cause) .build(); CouponDistributionTaskFailDO couponDistributionTaskFailDO = CouponDistributionTaskFailDO.builder() .batchId(taskDO.getBatchId()) .content(JSONUtil.toJsonStr(contentMap)) .build(); failRecordCacheList.add(couponDistributionTaskFailDO); if (failRecordCacheList.size() % 100 == 0) { couponDistributionFailMapper.insert(failRecordCacheList); failRecordCacheList.clear(); } syncProcess(processKey); } \/** * \u540c\u6b65\u6267\u884c\u8fdb\u5ea6\u5230\u7f13\u5b58 *\/ private void syncProcess(String processKey) { redisProxy.set(processKey, String.valueOf(rowRecord)); rowRecord++; }<\/code><\/p>\n\n\n\n<p><code>@Component @RequiredArgsConstructor @Slf4j @RocketMQMessageListener( consumerGroup = AdminConstantPool.COUPON_DISPENSE_CG, topic = AdminConstantPool.COUPON_DISPENSE_TOPIC, consumeMode = ConsumeMode.ORDERLY ) public class CouponDispenseConsumer implements RocketMQListener&lt;CouponDispenseMessage&gt; { private final RedisProxy redisProxy; private final CouponTemplateMapper couponTemplateMapper; private final UserCouponMapper userCouponMapper; private final CouponDistributionMapper couponDistributionMapper; @Override @Transactional(rollbackFor = Exception.class) public void onMessage(CouponDispenseMessage messageEvent) { log.info(\"[\u6d88\u8d39\u8005] \u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\uff0c\u6d88\u606f\u4f53\uff1a{}\", JSONUtil.toJsonStr(messageEvent)); \/\/ \u975e\u7ed3\u675f\u6807\u8bc6\u7684batchSetSize\u5df2\u7ecf\u5b58\u5728\uff0c\u53ef\u4ee5\u76f4\u63a5\u8fdb\u884c\u540e\u7eed\u903b\u8f91\u7684\u5904\u7406 if (!messageEvent.isEndFlag()) { handler(messageEvent); return; } \/\/ \u83b7\u53d6\u5f53\u524d\u7528\u6237\u9886\u5238\u96c6\u5408\u7684\u5927\u5c0f String batchUserReceiveCouponKey = StrUtil.join(\":\", \"user-coupon-collection\", messageEvent.getTaskId()); Long userReceiveCouponSetSize = redisProxy.getTemplateInstance().opsForSet().size(batchUserReceiveCouponKey); \/\/ \u7528\u6237\u9886\u5238\u96c6\u5408\u7f13\u5b58\u4e3a\u7a7a\u7684\u60c5\u51b5\uff1a1.\u7f13\u5b58\u5931\u6548 2.\u7f13\u5b58\u4e2d\u4f18\u60e0\u5238\u5e93\u5b58\u4e0d\u8db3 3.\u6700\u540e\u8fd9\u6279\u7528\u6237\u90fd\u5df2\u9886\u8fc7\u8be5\u4f18\u60e0\u5238 if (userReceiveCouponSetSize == null || userReceiveCouponSetSize == 0) { updateTaskStatus(messageEvent); return; } messageEvent.setBatchSetSize(userReceiveCouponSetSize.intValue()); handler(messageEvent); } \/** * \u5206\u53d1\u5238\u7684\u5904\u7406 *\/ @SneakyThrows private void handler(CouponDispenseMessage messageEvent) { Integer couponTemplateInventory = decrementInventory(messageEvent); if (couponTemplateInventory == 0) { RocketMQLogUtil.consumeFailure(\"\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\", messageEvent, \"\u4f18\u60e0\u5238\u5e93\u5b58\u4e3a\u7a7a\"); return; } String batchUserReceiveCouponKey = StrUtil.join(\":\", \"user-coupon-collection\", messageEvent.getTaskId()); \/\/ \u6263\u51cf\u4e86\u591a\u5c11\u5f20\u5e93\u5b58\u5c31\u4ece\u7528\u6237\u9886\u5238\u96c6\u5408\u4e2dpop\u51fa\u591a\u5c11\u4e2a\u5f85\u9886\u5238\u7684\u7528\u6237\u4fe1\u606f List&lt;String&gt; userAndRowMap = redisProxy.getTemplateInstance().opsForSet().pop(batchUserReceiveCouponKey, couponTemplateInventory); if (userAndRowMap == null) { RocketMQLogUtil.consumeFailure(\"\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\", messageEvent, \"\u7f13\u5b58\u4e2d\u7528\u6237\u9886\u5238\u96c6\u5408\u4e3a\u7a7a\"); return; } \/\/ \u521d\u59cb\u5316\u7528\u6237\u9886\u5238\u5b9e\u4f53\u96c6\u5408\u5e76\u6307\u5b9a\u5927\u5c0f\uff0c\u907f\u514dArrayList\u6269\u5bb9\u5f00\u9500 List&lt;UserCouponDO&gt; userCouponDOList = new ArrayList&lt;&gt;(couponTemplateInventory); userAndRowMap.forEach(item -&gt; { JSONObject recordObject = JSONUtil.parseObj(item); UserCouponDO userCouponDO = UserCouponDO.builder() .userId(recordObject.getLong(\"userId\")) .couponTemplateId(messageEvent.getCouponTemplateId()) .source(UserCouponSourceEnum.PLATFORM) \/\/ \u53ea\u6709\u5e73\u53f0\u624d\u6709\u6743\u9650\u83b7\u53d6\u7528\u6237\u9690\u79c1\u6570\u636e\u4ee5\u53ca\u5411\u7528\u6237\u6d3e\u53d1\u5238 .status(UserCouponStatusEnum.UNUSED) .receiveCount(1) \/\/ \u5e73\u53f0\u5238\u9650\u98861\u5f20 .build(); userCouponDOList.add(userCouponDO); }); \/\/ \u6279\u91cf\u65b0\u589e\u7528\u6237\u9886\u5238\u8bb0\u5f55 batchSaveUserCouponList(messageEvent, userCouponDOList); StringBuilder stringBuilder = new StringBuilder(); \/\/ \u5c06\u7528\u6237\u9886\u5238\u8bb0\u5f55\u4fdd\u5b58\u5230\u7f13\u5b58\u4e2d Map&lt;Object, Object&gt; valMap = MapUtil.builder() .put(\"count\", 1) \/\/ \u9886\u5238\u6b21\u6570 .put(\"date\", DateUtil.date()) \/\/ \u9886\u5238\u65f6\u95f4 .build(); userAndRowMap.forEach(each -&gt; { Long userId = JSONUtil.parseObj(each).getLong(\"userId\"); \/\/ \u53d6hashCode\u540e\u6309\u4f4d\u4e0e\u4fdd\u8bc1\u5176\u503c\u4e3a\u6b63 int mod = (userId.hashCode() &amp; Integer.MAX_VALUE) % 5; String key = stringBuilder.append(\"user-gained-coupon\").append(\":\").append(mod).toString(); stringBuilder.setLength(0); String hashKey = stringBuilder.append(userId).append(\"-\").append(messageEvent.getCouponTemplateId()).toString(); redisProxy.hashSet(key, hashKey, JSONUtil.toJsonStr(valMap)); stringBuilder.setLength(0); }); \/\/ \u786e\u4fdd\u5168\u90e8\u5206\u53d1\u5b8c\u6bd5\u540e\u66f4\u65b0\u63a8\u9001\u4efb\u52a1\u7684\u72b6\u6001 if (messageEvent.isEndFlag()) { updateTaskStatus(messageEvent); } } \/** * \u6263\u51cf\u4f18\u60e0\u5238\u5e93\u5b58 * @return \u6263\u51cf\u7684\u5e93\u5b58\u6570\u91cf *\/ private Integer decrementInventory(CouponDispenseMessage messageEvent) { int res = couponTemplateMapper.decrementCouponTemplateInventory(messageEvent.getCouponTemplateId(), messageEvent.getShopId(), messageEvent.getBatchSetSize()); \/\/ \u5e93\u5b58\u4e0d\u8db3\u5bfc\u81f4\u6263\u51cf\u5931\u8d25\uff0c\u9700\u8981\u91cd\u65b0\u83b7\u53d6\u53ef\u6263\u51cf\u7684\u5e93\u5b58\u91cf\uff0c\u5c3d\u91cf\u8ba9\u80fd\u53d1\u7684\u5238\u5206\u53d1\u5230\u7528\u6237 if (res == 0) { LambdaQueryWrapper&lt;CouponTemplateDO&gt; queryWrapper = Wrappers.lambdaQuery(CouponTemplateDO.class) .select(CouponTemplateDO::getInventory) .eq(CouponTemplateDO::getShopId, messageEvent.getShopId()) .eq(CouponTemplateDO::getId, messageEvent.getCouponTemplateId()); CouponTemplateDO couponTemplateDO = couponTemplateMapper.selectOne(queryWrapper); \/\/ \u82e5\u5e93\u5b58\u5df2\u7ecf\u4e3a0\u5219\u76f4\u63a5\u8fd4\u56de Integer inventory = couponTemplateDO.getInventory(); if (inventory == 0) return 0; \/\/ \u8fdb\u884c\u9012\u5f52\u6263\u51cf messageEvent.setBatchSetSize(inventory); return decrementInventory(messageEvent); } return messageEvent.getBatchSetSize(); } \/** * \u6279\u91cf\u65b0\u589e\u7528\u6237\u9886\u5238\u8bb0\u5f55 *\/ private void batchSaveUserCouponList(CouponDispenseMessage messageEvent, List&lt;UserCouponDO&gt; userCouponDOList) { try { userCouponMapper.insert(userCouponDOList); } catch (Exception ex) { Throwable cause = ex.getCause(); if (cause instanceof BatchExecutorException) { \/\/ \u6355\u83b7\u552f\u4e00\u7d22\u5f15\u5f02\u5e38\u5e76\u8bb0\u5f55\u65e5\u5fd7 RocketMQLogUtil.consumeFailure(\"\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\", messageEvent, ex); } throw new ServiceException(MerchantErrorCodeEnum.SERVICE_UPDATE_ERROR); } } \/** * \u66f4\u65b0\u63a8\u9001\u4efb\u52a1\u72b6\u6001 *\/ private void updateTaskStatus(CouponDispenseMessage messageEvent) { LambdaUpdateWrapper&lt;CouponDistributionTaskDO&gt; updateWrapper = Wrappers.lambdaUpdate(CouponDistributionTaskDO.class) .eq(CouponDistributionTaskDO::getId, messageEvent.getTaskId()) .set(CouponDistributionTaskDO::getStatus, CouponDistributionStatusEnum.SUCCESS); if (couponDistributionMapper.update(updateWrapper) == 1) { \/\/ \u6e05\u7406excel\u8bfb\u53d6\u8fdb\u5ea6\u7f13\u5b58 String processKey = StrUtil.join(\":\", \"coupon-distribution-process\", messageEvent.getTaskId()); redisProxy.remove(processKey); RocketMQLogUtil.consumeSuccess(\"\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\", messageEvent); return; } RocketMQLogUtil.consumeFailure(\"\u4f18\u60e0\u5238\u5206\u53d1\u5230\u7528\u6237\", messageEvent, \"\u66f4\u65b0\u63a8\u9001\u4efb\u52a1\u72b6\u6001\u5931\u8d25\"); }<\/code> \u200b<\/p>\n\n\n\n<hr class=\"wp-block-separator has-alpha-channel-opacity\"\/>\n\n\n\n<p>\u5047\u5982excel\u4e00\u5171\u67096000\u6761\u6570\u636e\uff0c\u626b\u63cf\u52305000\u7684\u65f6\u5019\u53d1\u9001\u6d88\u606f\u5230\u6d88\u606f\u961f\u5217\uff0c\u7136\u540e\u4e00\u76f4\u6267\u884c\u52306000\uff0c\u8fd9\u4e2d\u95f4\u7684\u5047\u5982\u6d88\u8d39\u8005\u8fd8\u6ca1\u5f00\u59cb\u8fdb\u884c\u6d88\u8d39\uff0c\u4e2d\u95f4\u4f1a\u8fde\u7eed\u4ea7\u751f1000\u6761\u6d88\u606f\u5230\u6d88\u8d39\u8005 \u4ed6\u4eec\u7684event\u8981\u6d88\u8017\u7684\u5e93\u5b58\u5206\u522b\u662f5000\uff0c5001\uff0c5002\uff0c\u3002\u3002\u30026000\u4ee5\u53cadoAfter\u4ea7\u751f\u76846000[\u8fd9\u91cc\u7684distributionEndFlag\u662ftrue]\u3002 \u7136\u540e\u5f00\u59cb\u6d88\u8d39\uff0c \u7b2c\u4e00\u6761\u6d88\u606f(5000)\u6b63\u5e38\u6263\u5e93\u5b585000\uff0c5001...5999\uff0c6000\u90fd\u4f1a\u56e0\u4e3aif (event.getDistributionEndFlag()) \u548cevent.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0\u88ab\u5254\u9664\uff0c\u4f46\u662f\u7531\u4e8edoAfter\u4ea7\u751f\u76846000\u7684endFlag\u662fTrue\uff0c\u6240\u4ee5\u4ed6\u4f1a\u901a\u8fc7\u7b2c\u4e8c\u56fe\u7684\u4ee3\u7801\uff0c\u7531\u4e8eInteger couponTemplateStock = decrementCouponTemplateStock(event, event.getBatchUserSetSize());\u56e0\u4e3a\u8fd9\u4e2a\u8fd4\u56de\u7684couponTemplateStock=Min(event\u4e2d\u8981\u6d88\u8017\u7684\u6570\u91cf=6000,\u6570\u636e\u5e93\u4f59\u989d) \uff0c\u5728\u8fd9\u4e00\u6b65 \u4f1a\u6d88\u8017\u5927\u91cf\u7684\u6570\u636e\u5e93\u5e93\u5b58\u5417\uff1f<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"395\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image-1024x395.png\" alt=\"\" class=\"wp-image-1304\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image-1024x395.png 1024w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image-300x116.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image-768x296.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image-1536x593.png 1536w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356404-image.png 1586w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/figure>\n\n\n\n<p>doAfter\u662f\u6839\u636e\u76ee\u524dredis\u4e2d\u5269\u4f59\u672a\u5206\u53d1\u7684\u4eba\u6570\u6765\u6263\u5e93\u5b58\u7684 \u6240\u4ee5\u4e0d\u4f1a\u5927\u6279\u91cf\u591a\u6263\u4e86<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"1024\" height=\"192\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356415-image-1024x192.png\" alt=\"\" class=\"wp-image-1305\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356415-image-1024x192.png 1024w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356415-image-300x56.png 300w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356415-image-768x144.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356415-image.png 1517w\" sizes=\"auto, (max-width: 1024px) 100vw, 1024px\" \/><\/figure>\n\n\n\n<p>\u4e00\u5f00\u59cb\u6ca1\u770b\u5230\u6700\u540e\uff0c\u540e\u9762\u5176\u5b9e\u8bf4\u4e86\u8981\u987a\u5e8f\u6d88\u606f\uff0c\u6240\u4ee5\u6839\u672c\u539f\u56e0\u5e94\u8be5\u662f\u56e0\u4e3a\u6d88\u606f\u662f\u987a\u5e8f\u6d88\u8d39\u7684\u5427\uff0c\u5982\u679c\u4e0d\u662f\u987a\u5e8f\u6d88\u8d39\u7684\uff0c\u4e0a\u4e00\u4e2a\u6279\u6b21\u76845000\u6761\u8fd8\u6ca1\u6709\u4eceredis\u5269\u4f59\u672a\u5206\u53d1\u7684\u4eba\u6570\u4e2dpop\u51fa\u6765\uff0c\u5047\u8bbe\u7b2c5001\u6761\u5c31\u5206\u53d1\u5b8c\u4e86\uff0c\u90a3\u4e48\u6263\u51cf\u7684\u5e93\u5b58\u6570\u91cf\u8fd8\u662f5001\uff0c\u5c31\u8fd8\u662f\u4f1a\u591a\u6263<\/p>\n\n\n\n<p>\u90a3 5001 \u5230 5999 \u8fd9\u4e9b\u662f\u4e0d\u662f\u90fd\u4f5c\u4e3a\u6d88\u606f\u53d1\u9001\u4e86\uff0c\u4f46\u662f\u90fd\u7531\u4e8eif (event.getDistributionEndFlag()) \u548cevent.getBatchUserSetSize() % BATCH_USER_COUPON_SIZE == 0\uff09\u88ab\u8fc7\u6ee4\u4e86\uff0c\u4e5f\u4e0d\u5c31\u662f\u8bf4\u8fd9\u4e9b\u90fd\u662f\u65e0\u7528\u7684\u6d88\u606f\u5417\uff0c\u8fd9\u4e0d\u6d6a\u8d39\u5417\uff0c\u6bcf 5 \u5343\u6761\u624d\u6709 1 \u6761\u6709\u7528\u7684\u6d88\u606f\uff1b\u6211\u89c9\u5f97\u8fd9\u91cc\u5e94\u8be5\u6539\u6210\u53d6\u6a21\u5224\u65ad\uff0c\u800c\u4e0d\u662f\u5c0f\u4e8e\uff0c\u8fd9\u6837\u5c31\u53ef\u4ee5\u907f\u514d\u4e86<\/p>\n\n\n\n<figure class=\"wp-block-image size-large\"><img loading=\"lazy\" decoding=\"async\" width=\"574\" height=\"1024\" src=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image-574x1024.png\" alt=\"\" class=\"wp-image-1306\" srcset=\"https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image-574x1024.png 574w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image-168x300.png 168w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image-768x1369.png 768w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image-861x1536.png 861w, https:\/\/eve2333.top\/wp-content\/uploads\/2025\/09\/1758356499-image.png 1060w\" sizes=\"auto, (max-width: 574px) 100vw, 574px\" \/><\/figure>\n","protected":false},"excerpt":{"rendered":"<p>&nbsp;\u7b2c16\u5c0f\u8282\uff1a\u7528\u6237\u67e5\u8be2\u4f18\u60e0\u5238\u4e4b\u7f13\u5b58\u51fb\u7a7f Git \u5206\u652f 20240826_dev_coupon-template-quer &#8230;<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"emotion":"","emotion_color":"","title_style":"","license":"","footnotes":""},"categories":[3],"tags":[20],"class_list":["post-1246","post","type-post","status-publish","format-standard","hentry","category-3","tag-20"],"_links":{"self":[{"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/posts\/1246","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/eve2333.top\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=1246"}],"version-history":[{"count":20,"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/posts\/1246\/revisions"}],"predecessor-version":[{"id":1393,"href":"https:\/\/eve2333.top\/index.php?rest_route=\/wp\/v2\/posts\/1246\/revisions\/1393"}],"wp:attachment":[{"href":"https:\/\/eve2333.top\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=1246"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/eve2333.top\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=1246"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/eve2333.top\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=1246"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}