{"id":67892,"date":"2025-05-10T12:06:36","date_gmt":"2025-05-10T04:06:36","guid":{"rendered":"https:\/\/fwq.ai\/blog\/67892\/"},"modified":"2025-05-10T12:06:36","modified_gmt":"2025-05-10T04:06:36","slug":"spring-boot%e5%9f%ba%e4%ba%8eredisson%e5%ae%9e%e7%8e%b0redis%e5%88%86%e5%b8%83%e5%bc%8f%e5%8f%af%e9%87%8d%e5%85%a5%e9%94%81%e6%ba%90%e7%a0%81%e8%af%a6%e8%a7%a3-2","status":"publish","type":"post","link":"https:\/\/fwq.ai\/blog\/67892\/","title":{"rendered":"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3"},"content":{"rendered":"<p><b><\/b> <\/p>\n<h1>Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3<\/h1>\n<p><span style=\"cursor: pointer\"><i><\/i>\u6536\u85cf<\/span> <\/p>\n<p>\u5927\u5bb6\u597d\uff0c\u6211\u4eec\u53c8\u89c1\u9762\u4e86\u554a~\u672c\u6587<span style=\"color: #FF6600;font-family:\">\u300aSpring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u300b<\/span>\u7684\u5185\u5bb9\u4e2d\u5c06\u4f1a\u6d89\u53ca\u5230<span style=\"color: #FF6600;font-family:\"><\/span>\u7b49\u7b49\u3002\u5982\u679c\u4f60\u6b63\u5728\u5b66\u4e60<span style=\"color: #FF6600;font-family:\">\u6570\u636e\u5e93<\/span>\u76f8\u5173\u77e5\u8bc6\uff0c\u6b22\u8fce\u5173\u6ce8\u6211\uff0c\u4ee5\u540e\u4f1a\u7ed9\u5927\u5bb6\u5e26\u6765\u66f4\u591a<span style=\"color: #FF6600;font-family:\">\u6570\u636e\u5e93<\/span>\u76f8\u5173\u6587\u7ae0\uff0c\u5e0c\u671b\u6211\u4eec\u80fd\u4e00\u8d77\u8fdb\u6b65\uff01\u4e0b\u9762\u5c31\u5f00\u59cb\u672c\u6587\u7684\u6b63\u5f0f\u5185\u5bb9~<\/p>\n<ul>\n<ul><\/ul>\n<\/ul>\n<ul>\n<ul><\/ul>\n<\/ul>\n<h3>\u4e00\u3001\u524d\u8a00<\/h3>\n<p>\u6211\u4eec\u5728\u5b9e\u73b0\u4f7f\u7528Redis\u5b9e\u73b0\u5206\u5e03\u5f0f\u9501\uff0c\u6700\u5f00\u59cb\u4e00\u822c\u4f7f\u7528<code>SET resource-name anystring NX EX max-lock-time<\/code>\u8fdb\u884c\u52a0\u9501\uff0c\u4f7f\u7528Lua\u811a\u672c\u4fdd\u8bc1\u539f\u5b50\u6027\u8fdb\u884c\u5b9e\u73b0\u91ca\u653e\u9501\u3002\u8fd9\u6837\u624b\u52a8\u5b9e\u73b0\u6bd4\u8f83\u9ebb\u70e6\uff0c\u5bf9\u6b64Redis\u5b98\u7f51\u4e5f\u660e\u786e\u8bf4Java\u7248\u4f7f\u7528<code>Redisson<\/code>\u6765\u5b9e\u73b0\u3002\u5c0f\u7f16\u4e5f\u662f\u770b\u4e86\u5b98\u7f51\u6162\u6162\u7684\u6478\u7d22\u6e05\u695a\uff0c\u7279\u5199\u6b64\u8bb0\u5f55\u4e00\u4e0b\u3002\u4ece\u5b98\u7f51\u5230\u6574\u5408Springboot\u5230\u6e90\u7801\u89e3\u8bfb\uff0c\u4ee5\u5355\u8282\u70b9\u4e3a\u4f8b\u3002<\/p>\n<h3>\u4e8c\u3001\u4e3a\u4ec0\u4e48\u4f7f\u7528Redisson<\/h3>\n<h4>1. \u6211\u4eec\u6253\u5f00\u5b98\u7f51<\/h4>\n<p>redis\u4e2d\u6587\u5b98\u7f51<\/p>\n<h4>2. \u6211\u4eec\u53ef\u4ee5\u770b\u5230\u5b98\u65b9\u8ba9\u6211\u4eec\u53bb\u4f7f\u7528\u5176\u4ed6<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2931cf.png\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe\" \/><\/p>\n<h4>3. \u6253\u5f00\u5b98\u65b9\u63a8\u8350<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c29db56.jpg\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe1\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe1\" \/><\/p>\n<h4>4. \u627e\u5230\u6587\u6863<\/h4>\n<p>Redisson\u5730\u5740<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2a78a1.jpg\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe2\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe2\" \/><\/p>\n<p>5. Redisson\u7ed3\u6784<\/p>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2b2469.png\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe3\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe3\" \/><\/p>\n<h3>\u4e09\u3001Springboot\u6574\u5408Redisson<\/h3>\n<h4>1. \u5bfc\u5165\u4f9d\u8d56<\/h4>\n<pre>\n   \n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     org.springframework.boot\n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     spring-boot-starter-data-redis\n    \n   \n   \n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     redis.clients\n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     jedis\n    \n   \n   \n   \n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     org.redisson\n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     redisson\n    \n&nbsp;&nbsp;&nbsp;&nbsp;\n    \n     3.12.0\n    \n   <\/pre>\n<h4>2. \u4ee5\u5b98\u7f51\u4e3a\u4f8b\u67e5\u770b\u5982\u4f55\u914d\u7f6e<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2bd56d.png\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe4\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe4\" \/><\/p>\n<h4>3. \u7f16\u5199\u914d\u7f6e\u7c7b<\/h4>\n<pre>import&nbsp;org.redisson.Redisson;\nimport&nbsp;org.redisson.api.RedissonClient;\nimport&nbsp;org.redisson.config.Config;\nimport&nbsp;org.springframework.context.annotation.Bean;\nimport&nbsp;org.springframework.context.annotation.Configuration;\n\n\/**\n&nbsp;*&nbsp;@author&nbsp;wangzhenjun\n&nbsp;*&nbsp;@date&nbsp;2022\/2\/9&nbsp;9:57\n&nbsp;*\/\n@Configuration\npublic&nbsp;class&nbsp;MyRedissonConfig&nbsp;{\n\n&nbsp;&nbsp;&nbsp;&nbsp;\/**\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*&nbsp;\u6240\u6709\u5bf9redisson\u7684\u4f7f\u7528\u90fd\u662f\u901a\u8fc7RedissonClient\u6765\u64cd\u4f5c\u7684\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*&nbsp;@return\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;*\/\n&nbsp;&nbsp;&nbsp;&nbsp;@Bean(destroyMethod=\"shutdown\")\n&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;RedissonClient&nbsp;redisson(){\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;1.&nbsp;\u521b\u5efa\u914d\u7f6e\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Config&nbsp;config&nbsp;=&nbsp;new&nbsp;Config();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u4e00\u5b9a\u8981\u52a0redis:\/\/\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;config.useSingleServer().setAddress(\"redis:\/\/192.168.17.130:6379\");\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;2.&nbsp;\u6839\u636econfig\u521b\u5efa\u51faredissonClient\u5b9e\u4f8b\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;RedissonClient&nbsp;redissonClient&nbsp;=&nbsp;Redisson.create(config);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;redissonClient;\n&nbsp;&nbsp;&nbsp;&nbsp;}\n}<\/pre>\n<h4>4. \u5b98\u7f51\u6d4b\u8bd5\u52a0\u9501\u4f8b\u5b50<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2c8262.jpg\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe5\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe5\" \/><\/p>\n<h4>5. \u6839\u636e\u5b98\u7f51\u7b80\u5355Controller\u63a5\u53e3\u7f16\u5199<\/h4>\n<pre>@ResponseBody\n@GetMapping(\"\/hello\")\npublic&nbsp;String&nbsp;hello(){\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;1.\u83b7\u53d6\u4e00\u628a\u9501\uff0c\u53ea\u8981\u9501\u540d\u5b57\u4e00\u6837\uff0c\u5c31\u662f\u540c\u4e00\u628a\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;RLock&nbsp;lock&nbsp;=&nbsp;redisson.getLock(\"my-lock\");\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;2.&nbsp;\u52a0\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;lock.lock();\/\/&nbsp;\u963b\u585e\u8bd5\u7b49\u5f85&nbsp;&nbsp;\u9ed8\u8ba4\u52a0\u7684\u90fd\u662f30s\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u5e26\u53c2\u6570\u60c5\u51b5\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;lock.lock(10,&nbsp;TimeUnit.SECONDS);\/\/&nbsp;10s\u81ea\u52a8\u89e3\u9501\uff0c\u81ea\u52a8\u89e3\u9501\u65f6\u95f4\u4e00\u5b9a\u8981\u5927\u4e8e\u4e1a\u52a1\u7684\u6267\u884c\u65f6\u95f4\u3002\n&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;System.out.println(\"\u52a0\u9501\u6210\u529f\"&nbsp;+&nbsp;Thread.currentThread().getId());\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Thread.sleep(30000);\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;catch&nbsp;(InterruptedException&nbsp;e)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;e.printStackTrace();\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;finally&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;3.&nbsp;\u89e3\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;System.out.println(\"\u89e3\u9501\u6210\u529f\uff1a\"&nbsp;+&nbsp;Thread.currentThread().getId());\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;lock.unlock();\n&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;\"hello\";\n}<\/pre>\n<h4>6. \u6d4b\u8bd5<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2d2778.jpg\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe6\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe6\" \/><\/p>\n<h3>\u56db\u3001lock.lock()\u6e90\u7801\u5206\u6790<\/h3>\n<h4>1. \u6253\u5f00RedissonLock\u5b9e\u73b0\u7c7b<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2de5bd.png\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe7\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe7\" \/><\/p>\n<h4>2. \u627e\u5230\u5b9e\u73b0\u65b9\u6cd5<\/h4>\n<pre>@Override\npublic&nbsp;void&nbsp;lock()&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u6211\u4eec\u53d1\u73b0\u4e0d\u7a7f\u8fc7\u671f\u65f6\u95f4\u6e90\u7801\u9ed8\u8ba4\u8fc7\u671f\u65f6\u95f4\u4e3a-1\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;lock(-1,&nbsp;null,&nbsp;false);\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;catch&nbsp;(InterruptedException&nbsp;e)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;new&nbsp;IllegalStateException();\n&nbsp;&nbsp;&nbsp;&nbsp;}\n}<\/pre>\n<h4>3. \u6309\u4f4fCtrl\u8fdb\u53bblock\u65b9\u6cd5<\/h4>\n<pre>private&nbsp;void&nbsp;lock(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;boolean&nbsp;interruptibly)&nbsp;throws&nbsp;InterruptedException&nbsp;{\n\t\/\/&nbsp;\u83b7\u53d6\u7ebf\u7a0b\u7684id\uff0c\u5360\u6709\u9501\u7684\u65f6\u5019field\u7684\u503c\u4e3aUUID:\u7ebf\u7a0b\u53f7id\n&nbsp;&nbsp;&nbsp;&nbsp;long&nbsp;threadId&nbsp;=&nbsp;Thread.currentThread().getId();\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u5c1d\u8bd5\u83b7\u5f97\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;Long&nbsp;ttl&nbsp;=&nbsp;tryAcquire(leaseTime,&nbsp;unit,&nbsp;threadId);\n&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;lock&nbsp;acquired&nbsp;\u83b7\u5f97\u9501\uff0c\u8fd4\u56de\n&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ttl&nbsp;==&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n&nbsp;&nbsp;&nbsp;&nbsp;}\n\t\/\/&nbsp;\u8fd9\u91cc\u8bf4\u660e\u83b7\u53d6\u9501\u5931\u8d25\uff0c\u5c31\u901a\u8fc7\u7ebf\u7a0bid\u8ba2\u9605\u8fd9\u4e2a\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;RFuture\n   \n    &nbsp;future&nbsp;=&nbsp;subscribe(threadId);\n&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(interruptibly)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;commandExecutor.syncSubscriptionInterrupted(future);\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;else&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;commandExecutor.syncSubscription(future);\n&nbsp;&nbsp;&nbsp;&nbsp;}\n\n&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u8fd9\u91cc\u8fdb\u884c\u81ea\u65cb\uff0c\u4e0d\u65ad\u5c1d\u8bd5\u83b7\u53d6\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;while&nbsp;(true)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u7ee7\u7eed\u5c1d\u8bd5\u83b7\u53d6\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ttl&nbsp;=&nbsp;tryAcquire(leaseTime,&nbsp;unit,&nbsp;threadId);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;lock&nbsp;acquired&nbsp;\u83b7\u53d6\u6210\u529f\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ttl&nbsp;==&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u76f4\u63a5\u8fd4\u56de\uff0c\u6311\u51fa\u81ea\u65cb\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;break;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;waiting&nbsp;for&nbsp;message&nbsp;\u7ee7\u7eed\u7b49\u5f85\u83b7\u5f97\u9501\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ttl&nbsp;&gt;=&nbsp;0)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.getNow().getLatch().tryAcquire(ttl,&nbsp;TimeUnit.MILLISECONDS);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;catch&nbsp;(InterruptedException&nbsp;e)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(interruptibly)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;e;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.getNow().getLatch().tryAcquire(ttl,&nbsp;TimeUnit.MILLISECONDS);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;else&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(interruptibly)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.getNow().getLatch().acquire();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;else&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.getNow().getLatch().acquireUninterruptibly();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;finally&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u53d6\u6d88\u8ba2\u9605\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;unsubscribe(future,&nbsp;threadId);\n&nbsp;&nbsp;&nbsp;&nbsp;}\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;get(lockAsync(leaseTime,&nbsp;unit));\n}\n   <\/pre>\n<h4>4. \u8fdb\u53bb\u5c1d\u8bd5\u83b7\u53d6\u9501\u65b9\u6cd5<\/h4>\n<p><img decoding=\"async\" src=\"https:\/\/www.17golang.com\/uploads\/20230507\/1683460802645792c2e7e6d.png\" class=\"aligncenter\" title=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe8\" alt=\"Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u63d2\u56fe8\" \/><\/p>\n<pre>private&nbsp;Long&nbsp;tryAcquire(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;long&nbsp;threadId)&nbsp;{\n\t\/\/&nbsp;\u76f4\u63a5\u8fdb\u5165\u5f02\u6b65\u65b9\u6cd5\n&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;get(tryAcquireAsync(leaseTime,&nbsp;unit,&nbsp;threadId));\n}\n\nprivate&nbsp;\n   \n    &nbsp;RFuture\n    \n     &nbsp;tryAcquireAsync(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;long&nbsp;threadId)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u8fd9\u91cc\u8fdb\u884c\u5224\u65ad\u5982\u679c\u6ca1\u6709\u8bbe\u7f6e\u53c2\u6570leaseTime&nbsp;=&nbsp;-1 &nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(leaseTime&nbsp;!=&nbsp;-1)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;tryLockInnerAsync(leaseTime,&nbsp;unit,&nbsp;threadId,&nbsp;RedisCommands.EVAL_LONG); &nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u6b64\u65b9\u6cd5\u8fdb\u884c\u83b7\u5f97\u9501\uff0c\u8fc7\u671f\u65f6\u95f4\u4e3a\u770b\u95e8\u72d7\u7684\u9ed8\u8ba4\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;private&nbsp;long&nbsp;lockWatchdogTimeout&nbsp;=&nbsp;30&nbsp;*&nbsp;1000;\u770b\u95e8\u72d7\u9ed8\u8ba4\u8fc7\u671f\u65f6\u95f4\u4e3a30s &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u52a0\u9501\u548c\u8fc7\u671f\u65f6\u95f4\u8981\u4fdd\u8bc1\u539f\u5b50\u6027\uff0c\u8fd9\u4e2a\u65b9\u6cd5\u540e\u9762\u80af\u5b9a\u8c03\u7528\u6267\u884c\u4e86Lua\u811a\u672c\uff0c\u6211\u4eec\u4e0b\u9762\u5728\u770b &nbsp;&nbsp;&nbsp;&nbsp;RFuture\n     \n      &nbsp;ttlRemainingFuture&nbsp;=&nbsp;tryLockInnerAsync(commandExecutor.getConnectionManager().getCfg().getLockWatchdogTimeout(),&nbsp;TimeUnit.MILLISECONDS,&nbsp;threadId,&nbsp;RedisCommands.EVAL_LONG); &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u5f00\u542f\u4e00\u4e2a\u5b9a\u65f6\u4efb\u52a1\u8fdb\u884c\u4e0d\u65ad\u5237\u65b0\u8fc7\u671f\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;ttlRemainingFuture.onComplete((ttlRemaining,&nbsp;e)&nbsp;-&gt;&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(e&nbsp;!=&nbsp;null)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;lock&nbsp;acquired&nbsp;\u83b7\u5f97\u9501 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ttlRemaining&nbsp;==&nbsp;null)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u5237\u65b0\u8fc7\u671f\u65f6\u95f4\u65b9\u6cd5\uff0c\u6211\u4eec\u4e0b\u4e00\u6b65\u8be6\u7ec6\u8bf4\u4e00\u4e0b &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;scheduleExpirationRenewal(threadId); &nbsp;&nbsp;&nbsp;&nbsp;}); &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;ttlRemainingFuture;\n     \n    \n   <\/pre>\n<h4>5. \u67e5\u770btryLockInnerAsync()\u65b9\u6cd5<\/h4>\n<pre>\n   \n    &nbsp;RFuture\n    \n     &nbsp;tryLockInnerAsync(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;long&nbsp;threadId,&nbsp;RedisStrictCommand\n     \n      &nbsp;command)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;internalLockLeaseTime&nbsp;=&nbsp;unit.toMillis(leaseTime); &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;commandExecutor.evalWriteAsync(getName(),&nbsp;LongCodec.INSTANCE,&nbsp;command, &nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;\/\/&nbsp;\u9996\u5148\u5224\u65ad\u9501\u662f\u5426\u5b58\u5728 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(redis.call('exists',&nbsp;KEYS[1])&nbsp;==&nbsp;0)&nbsp;then&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u5b58\u5728\u5219\u83b7\u53d6\u9501 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('hset',&nbsp;KEYS[1],&nbsp;ARGV[2],&nbsp;1);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u7136\u540e\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('pexpire',&nbsp;KEYS[1],&nbsp;ARGV[1]);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;hexists\u67e5\u770b\u54c8\u5e0c\u8868\u7684\u6307\u5b9a\u5b57\u6bb5\u662f\u5426\u5b58\u5728\uff0c\u5b58\u5728\u9501\u5e76\u4e14\u662f\u5f53\u524d\u7ebf\u7a0b\u6301\u6709\u9501 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(redis.call('hexists',&nbsp;KEYS[1],&nbsp;ARGV[2])&nbsp;==&nbsp;1)&nbsp;then&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;hincrby\u81ea\u589e\u4e00 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('hincrby',&nbsp;KEYS[1],&nbsp;ARGV[2],&nbsp;1);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u9501\u7684\u503c\u5927\u4e8e1\uff0c\u8bf4\u660e\u662f\u53ef\u91cd\u5165\u9501\uff0c\u91cd\u7f6e\u8fc7\u671f\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('pexpire',&nbsp;KEYS[1],&nbsp;ARGV[1]);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u9501\u5df2\u5b58\u5728\uff0c\u4e14\u4e0d\u662f\u672c\u7ebf\u7a0b\uff0c\u5219\u8fd4\u56de\u8fc7\u671f\u65f6\u95f4ttl &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;redis.call('pttl',&nbsp;KEYS[1]);\", &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Collections.\n      singletonList(getName()),&nbsp;internalLockLeaseTime,&nbsp;getLockName(threadId)); }\n     \n    \n   <\/pre>\n<h4>6. \u8fdb\u51654\u7559\u4e0b\u7684\u5b9a\u65f6\u4efb\u52a1scheduleExpirationRenewal()\u65b9\u6cd5<\/h4>\n<p>\u4e00\u6b65\u6b65\u5f80\u4e0b\u627e\u6e90\u7801\uff1ascheduleExpirationRenewal &#8212;&gt;renewExpiration<\/p>\n<p>\u6839\u636e\u4e0b\u9762\u6e90\u7801\uff0c\u5b9a\u65f6\u4efb\u52a1\u5237\u65b0\u65f6\u95f4\u4e3a\uff1ainternalLockLeaseTime \/ 3\uff0c\u662f\u770b\u95e8\u72d7\u76841\/3\uff0c\u5373\u4e3a10s\u5237\u65b0\u4e00\u6b21<\/p>\n<pre>private&nbsp;void&nbsp;renewExpiration()&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;ExpirationEntry&nbsp;ee&nbsp;=&nbsp;EXPIRATION_RENEWAL_MAP.get(getEntryName());\n&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ee&nbsp;==&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;\n&nbsp;&nbsp;&nbsp;&nbsp;Timeout&nbsp;task&nbsp;=&nbsp;commandExecutor.getConnectionManager().newTimeout(new&nbsp;TimerTask()&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;@Override\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;void&nbsp;run(Timeout&nbsp;timeout)&nbsp;throws&nbsp;Exception&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ExpirationEntry&nbsp;ent&nbsp;=&nbsp;EXPIRATION_RENEWAL_MAP.get(getEntryName());\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(ent&nbsp;==&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Long&nbsp;threadId&nbsp;=&nbsp;ent.getFirstThreadId();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(threadId&nbsp;==&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;RFuture\n   \n    &nbsp;future&nbsp;=&nbsp;renewExpirationAsync(threadId);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.onComplete((res,&nbsp;e)&nbsp;-&gt;&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(e&nbsp;!=&nbsp;null)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;log.error(\"Can't&nbsp;update&nbsp;lock&nbsp;\"&nbsp;+&nbsp;getName()&nbsp;+&nbsp;\"&nbsp;expiration\",&nbsp;e);\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(res)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;reschedule&nbsp;itself\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;renewExpiration();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;});\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;},&nbsp;internalLockLeaseTime&nbsp;\/&nbsp;3,&nbsp;TimeUnit.MILLISECONDS);\n&nbsp;&nbsp;&nbsp;&nbsp;\n&nbsp;&nbsp;&nbsp;&nbsp;ee.setTimeout(task);\n}\n   <\/pre>\n<h3>\u4e94\u3001lock.lock(10, TimeUnit.SECONDS)\u6e90\u7801\u5206\u6790<\/h3>\n<p>1. \u6253\u5f00\u5b9e\u73b0\u7c7b<\/p>\n<pre>@Override\npublic&nbsp;void&nbsp;lock(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u8fd9\u91cc\u7684\u8fc7\u671f\u65f6\u95f4\u4e3a\u6211\u4eec\u8f93\u5165\u768410\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;lock(leaseTime,&nbsp;unit,&nbsp;false);\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;catch&nbsp;(InterruptedException&nbsp;e)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;new&nbsp;IllegalStateException();\n&nbsp;&nbsp;&nbsp;&nbsp;}\n}<\/pre>\n<p>2. \u65b9\u6cd5<code>lock()<\/code>\u5b9e\u73b0\u5c55\u793a\uff0c\u540c\u4e09.3\u6e90\u7801<\/p>\n<p>3. \u76f4\u63a5\u6765\u5230\u5c1d\u8bd5\u83b7\u5f97\u9501<code>tryAcquireAsync()<\/code>\u65b9\u6cd5<\/p>\n<pre>private&nbsp;\n   \n    &nbsp;RFuture\n    \n     &nbsp;tryAcquireAsync(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;long&nbsp;threadId)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u8fd9\u91cc\u8fdb\u884c\u5224\u65ad\u5982\u679c\u6ca1\u6709\u8bbe\u7f6e\u53c2\u6570leaseTime&nbsp;=&nbsp;-1,\u6b64\u65f6\u6211\u4eec\u4e3a10 &nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(leaseTime&nbsp;!=&nbsp;-1)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u6765\u5230\u6b64\u65b9\u6cd5 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;tryLockInnerAsync(leaseTime,&nbsp;unit,&nbsp;threadId,&nbsp;RedisCommands.EVAL_LONG); &nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u6b64\u5904\u7701\u7565\u540e\u9762\u5185\u5bb9\uff0c\u524d\u9762\u4ee5\u8be6\u7ec6\u8bf4\u660e\u3002\u3002\u3002\u3002 }\n    \n   <\/pre>\n<p>4. \u6253\u5f00<code>tryLockInnerAsync()<\/code>\u65b9\u6cd5<\/p>\n<p>\u6211\u4eec\u4e0d\u96be\u53d1\u73b0\u548c\u6ca1\u6709\u4f20\u8fc7\u671f\u65f6\u95f4\u7684\u65b9\u6cd5\u4e00\u6837\uff0c\u53ea\u4e0d\u8fc7leaseTime\u7684\u503c\u53d8\u4e86\u3002<\/p>\n<pre>\n   \n    &nbsp;RFuture\n    \n     &nbsp;tryLockInnerAsync(long&nbsp;leaseTime,&nbsp;TimeUnit&nbsp;unit,&nbsp;long&nbsp;threadId,&nbsp;RedisStrictCommand\n     \n      &nbsp;command)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;internalLockLeaseTime&nbsp;=&nbsp;unit.toMillis(leaseTime); &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;commandExecutor.evalWriteAsync(getName(),&nbsp;LongCodec.INSTANCE,&nbsp;command, &nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;\/\/&nbsp;\u9996\u5148\u5224\u65ad\u9501\u662f\u5426\u5b58\u5728 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(redis.call('exists',&nbsp;KEYS[1])&nbsp;==&nbsp;0)&nbsp;then&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u5b58\u5728\u5219\u83b7\u53d6\u9501 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('hset',&nbsp;KEYS[1],&nbsp;ARGV[2],&nbsp;1);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u7136\u540e\u8bbe\u7f6e\u8fc7\u671f\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('pexpire',&nbsp;KEYS[1],&nbsp;ARGV[1]);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;hexists\u67e5\u770b\u54c8\u5e0c\u8868\u7684\u6307\u5b9a\u5b57\u6bb5\u662f\u5426\u5b58\u5728\uff0c\u5b58\u5728\u9501\u5e76\u4e14\u662f\u5f53\u524d\u7ebf\u7a0b\u6301\u6709\u9501 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(redis.call('hexists',&nbsp;KEYS[1],&nbsp;ARGV[2])&nbsp;==&nbsp;1)&nbsp;then&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;hincrby\u81ea\u589e\u4e00 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('hincrby',&nbsp;KEYS[1],&nbsp;ARGV[2],&nbsp;1);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u9501\u7684\u503c\u5927\u4e8e1\uff0c\u8bf4\u660e\u662f\u53ef\u91cd\u5165\u9501\uff0c\u91cd\u7f6e\u8fc7\u671f\u65f6\u95f4 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('pexpire',&nbsp;KEYS[1],&nbsp;ARGV[1]);&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u9501\u5df2\u5b58\u5728\uff0c\u4e14\u4e0d\u662f\u672c\u7ebf\u7a0b\uff0c\u5219\u8fd4\u56de\u8fc7\u671f\u65f6\u95f4ttl &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;redis.call('pttl',&nbsp;KEYS[1]);\", &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Collections.\n      singletonList(getName()),&nbsp;internalLockLeaseTime,&nbsp;getLockName(threadId)); }\n     \n    \n   <\/pre>\n<h3>\u516d\u3001lock.unlock()\u6e90\u7801\u5206\u6790<\/h3>\n<p>1. \u6253\u5f00\u65b9\u6cd5\u5b9e\u73b0<\/p>\n<pre>@Override\npublic&nbsp;void&nbsp;unlock()&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;try&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u70b9\u51fb\u8fdb\u5165\u91ca\u653e\u9501\u65b9\u6cd5\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;get(unlockAsync(Thread.currentThread().getId()));\n&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;catch&nbsp;(RedisException&nbsp;e)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(e.getCause()&nbsp;instanceof&nbsp;IllegalMonitorStateException)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;(IllegalMonitorStateException)&nbsp;e.getCause();\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;else&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;e;\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;}\n&nbsp;&nbsp;&nbsp;&nbsp;\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Future\n   \n    &nbsp;future&nbsp;=&nbsp;unlockAsync();\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;future.awaitUninterruptibly();\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(future.isSuccess())&nbsp;{\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return;\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(future.cause()&nbsp;instanceof&nbsp;IllegalMonitorStateException)&nbsp;{\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;(IllegalMonitorStateException)future.cause();\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}\n\/\/&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;throw&nbsp;commandExecutor.convertException(future);\n}\n   <\/pre>\n<p>2. \u6253\u5f00<code>unlockAsync()<\/code>\u65b9\u6cd5<\/p>\n<pre>@Override\npublic&nbsp;RFuture\n   \n    &nbsp;unlockAsync(long&nbsp;threadId)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;RPromise\n    \n     &nbsp;result&nbsp;=&nbsp;new&nbsp;RedissonPromise\n     \n      (); &nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u89e3\u9501\u65b9\u6cd5\uff0c\u540e\u9762\u5c55\u5f00\u8bf4 &nbsp;&nbsp;&nbsp;&nbsp;RFuture\n      \n       &nbsp;future&nbsp;=&nbsp;unlockInnerAsync(threadId); \/\/&nbsp;\u5b8c\u6210 &nbsp;&nbsp;&nbsp;&nbsp;future.onComplete((opStatus,&nbsp;e)&nbsp;-&gt;&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(e&nbsp;!=&nbsp;null)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; \/\/&nbsp;\u53d6\u6d88\u5230\u671f\u7eed\u8ba2 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;cancelExpirationRenewal(threadId); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u5c06\u8fd9\u4e2a\u672a\u6765\u6807\u8bb0\u4e3a\u5931\u8d25\u5e76\u901a\u77e5\u6240\u6709\u4eba &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;result.tryFailure(e); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;} \/\/&nbsp;\u72b6\u6001\u4e3a\u7a7a\uff0c\u8bf4\u660e\u89e3\u9501\u7684\u7ebf\u7a0b\u548c\u5f53\u524d\u9501\u4e0d\u662f\u540c\u4e00\u4e2a\u7ebf\u7a0b &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(opStatus&nbsp;==&nbsp;null)&nbsp;{ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;IllegalMonitorStateException&nbsp;cause&nbsp;=&nbsp;new&nbsp;IllegalMonitorStateException(\"attempt&nbsp;to&nbsp;unlock&nbsp;lock,&nbsp;not&nbsp;locked&nbsp;by&nbsp;current&nbsp;thread&nbsp;by&nbsp;node&nbsp;id:&nbsp;\" &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;+&nbsp;id&nbsp;+&nbsp;\"&nbsp;thread-id:&nbsp;\"&nbsp;+&nbsp;threadId); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;result.tryFailure(cause); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;} &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;cancelExpirationRenewal(threadId); &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;result.trySuccess(null); &nbsp;&nbsp;&nbsp;&nbsp;}); &nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;result; }\n      \n     \n    \n   <\/pre>\n<p>3. \u6253\u5f00<code>unlockInnerAsync()<\/code>\u65b9\u6cd5<\/p>\n<pre>protected&nbsp;RFuture\n   \n    &nbsp;unlockInnerAsync(long&nbsp;threadId)&nbsp;{\n&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;commandExecutor.evalWriteAsync(getName(),&nbsp;LongCodec.INSTANCE,&nbsp;RedisCommands.EVAL_BOOLEAN,\n&nbsp;&nbsp;&nbsp;&nbsp;\t\t\/\/&nbsp;\u5224\u65ad\u91ca\u653e\u9501\u7684\u7ebf\u7a0b\u548c\u5df2\u5b58\u5728\u9501\u7684\u7ebf\u7a0b\u662f\u4e0d\u662f\u540c\u4e00\u4e2a\u7ebf\u7a0b\uff0c\u4e0d\u662f\u8fd4\u56de\u7a7a\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(redis.call('hexists',&nbsp;KEYS[1],&nbsp;ARGV[3])&nbsp;==&nbsp;0)&nbsp;then&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u91ca\u653e\u9501\u540e\uff0c\u52a0\u9501\u6b21\u6570\u51cf\u4e00\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"local&nbsp;counter&nbsp;=&nbsp;redis.call('hincrby',&nbsp;KEYS[1],&nbsp;ARGV[3],&nbsp;-1);&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\/\/&nbsp;\u5224\u65ad\u5269\u4f59\u6570\u91cf\u662f\u5426\u5927\u4e8e0\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"if&nbsp;(counter&nbsp;&gt;&nbsp;0)&nbsp;then&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u5927\u4e8e0&nbsp;\uff0c\u5219\u5237\u65b0\u8fc7\u671f\u65f6\u95f4\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('pexpire',&nbsp;KEYS[1],&nbsp;ARGV[2]);&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;0;&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"else&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\t\/\/&nbsp;\u91ca\u653e\u9501\uff0c\u5220\u9664key\u5e76\u53d1\u5e03\u9501\u91ca\u653e\u7684\u6d88\u606f\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('del',&nbsp;KEYS[1]);&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"redis.call('publish',&nbsp;KEYS[2],&nbsp;ARGV[1]);&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;1;&nbsp;\"+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"end;&nbsp;\"&nbsp;+\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;\"return&nbsp;nil;\",\n&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Arrays.\n    asList(getName(),&nbsp;getChannelName()),&nbsp;LockPubSub.UNLOCK_MESSAGE,&nbsp;internalLockLeaseTime,&nbsp;getLockName(threadId)); }\n   <\/pre>\n","protected":false},"excerpt":{"rendered":"<p>Spring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3 \u6536\u85cf \u5927\u5bb6\u597d\uff0c\u6211\u4eec\u53c8\u89c1\u9762\u4e86\u554a~\u672c\u6587\u300aSpring Boot\u57fa\u4e8eRedisson\u5b9e\u73b0Redis\u5206\u5e03\u5f0f\u53ef\u91cd\u5165\u9501\u6e90\u7801\u8be6\u89e3\u300b\u7684\u5185\u5bb9\u4e2d\u5c06\u4f1a\u6d89\u53ca\u5230\u7b49\u7b49\u3002\u5982\u679c\u4f60\u6b63\u5728\u5b66\u4e60\u6570\u636e\u5e93\u76f8\u5173\u77e5\u8bc6\uff0c\u6b22\u8fce\u5173\u6ce8\u6211\uff0c\u4ee5\u540e\u4f1a\u7ed9\u5927\u5bb6\u5e26\u6765\u66f4\u591a\u6570\u636e\u5e93\u76f8\u5173\u6587\u7ae0\uff0c\u5e0c\u671b\u6211\u4eec\u80fd\u4e00\u8d77\u8fdb\u6b65\uff01\u4e0b\u9762\u5c31\u5f00\u59cb\u672c\u6587\u7684\u6b63\u5f0f\u5185\u5bb9~ \u4e00\u3001\u524d\u8a00 \u6211\u4eec\u5728\u5b9e\u73b0\u4f7f\u7528Redis\u5b9e\u73b0\u5206\u5e03\u5f0f\u9501\uff0c\u6700\u5f00\u59cb\u4e00\u822c\u4f7f\u7528SET resource-name anystring NX EX max-lock-time\u8fdb\u884c\u52a0\u9501\uff0c\u4f7f\u7528Lua\u811a\u672c\u4fdd\u8bc1\u539f\u5b50\u6027\u8fdb\u884c\u5b9e\u73b0\u91ca\u653e\u9501\u3002\u8fd9\u6837\u624b\u52a8\u5b9e\u73b0\u6bd4\u8f83\u9ebb\u70e6\uff0c\u5bf9\u6b64Redis\u5b98\u7f51\u4e5f\u660e\u786e\u8bf4Java\u7248\u4f7f\u7528Redisson\u6765\u5b9e\u73b0\u3002\u5c0f\u7f16\u4e5f\u662f\u770b\u4e86\u5b98\u7f51\u6162\u6162\u7684\u6478\u7d22\u6e05\u695a\uff0c\u7279\u5199\u6b64\u8bb0\u5f55\u4e00\u4e0b\u3002\u4ece\u5b98\u7f51\u5230\u6574\u5408Springboot\u5230\u6e90\u7801\u89e3\u8bfb\uff0c\u4ee5\u5355\u8282\u70b9\u4e3a\u4f8b\u3002 \u4e8c\u3001\u4e3a\u4ec0\u4e48\u4f7f\u7528Redisson 1. \u6211\u4eec\u6253\u5f00\u5b98\u7f51 redis\u4e2d\u6587\u5b98\u7f51 2. \u6211\u4eec\u53ef\u4ee5\u770b\u5230\u5b98\u65b9\u8ba9\u6211\u4eec\u53bb\u4f7f\u7528\u5176\u4ed6 3. \u6253\u5f00\u5b98\u65b9\u63a8\u8350 4. \u627e\u5230\u6587\u6863 Redisson\u5730\u5740 5. Redisson\u7ed3\u6784 \u4e09\u3001Springboot\u6574\u5408Redisson 1. \u5bfc\u5165\u4f9d\u8d56 &nbsp;&nbsp;&nbsp;&nbsp; org.springframework.boot &nbsp;&nbsp;&nbsp;&nbsp; spring-boot-starter-data-redis &nbsp;&nbsp;&nbsp;&nbsp; redis.clients &nbsp;&nbsp;&nbsp;&nbsp; jedis &nbsp;&nbsp;&nbsp;&nbsp; org.redisson &nbsp;&nbsp;&nbsp;&nbsp; redisson &nbsp;&nbsp;&nbsp;&nbsp; 3.12.0 2. \u4ee5\u5b98\u7f51\u4e3a\u4f8b\u67e5\u770b\u5982\u4f55\u914d\u7f6e 3. \u7f16\u5199\u914d\u7f6e\u7c7b import&nbsp;org.redisson.Redisson; import&nbsp;org.redisson.api.RedissonClient; import&nbsp;org.redisson.config.Config; import&nbsp;org.springframework.context.annotation.Bean; import&nbsp;org.springframework.context.annotation.Configuration; \/** &nbsp;*&nbsp;@author&nbsp;wangzhenjun &nbsp;*&nbsp;@date&nbsp;2022\/2\/9&nbsp;9:57 &nbsp;*\/ [&hellip;]<\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"closed","ping_status":"","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[8],"tags":[],"class_list":["post-67892","post","type-post","status-publish","format-standard","hentry","category-os"],"_links":{"self":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts\/67892","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/comments?post=67892"}],"version-history":[{"count":0,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/posts\/67892\/revisions"}],"wp:attachment":[{"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/media?parent=67892"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/categories?post=67892"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/fwq.ai\/blog\/wp-json\/wp\/v2\/tags?post=67892"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}