2 Commits 5dad413607 ... 50346a7742

Author SHA1 Message Date
  hyg 50346a7742 Merge branch 'master' of http://172.16.90.201:3000/SharedLibrary-SystemCode/UtilsDemo 4 years ago
  hyg 2f8723b4a7 auto commit 4 years ago
81 changed files with 7572 additions and 10 deletions
  1. 103 0
      UtilsDemo-Excel/UtilsDemo-Excel.iml
  2. 48 0
      UtilsDemo-Excel/pom.xml
  3. 57 0
      UtilsDemo-SecureEncryption/UtilsDemo-SecureEncryption.iml
  4. 2 0
      UtilsDemo-WX/.gitignore
  5. 140 0
      UtilsDemo-WX/UtilsDemo-WX.iml
  6. 145 0
      UtilsDemo-WX/pom.xml
  7. 19 0
      UtilsDemo-WX/src/main/java/com/example/WXApplication.java
  8. 185 0
      UtilsDemo-WX/src/main/java/com/example/base/BaseController.java
  9. 9 0
      UtilsDemo-WX/src/main/java/com/example/base/BaseEntity.java
  10. 32 0
      UtilsDemo-WX/src/main/java/com/example/base/ResponseBase.java
  11. 87 0
      UtilsDemo-WX/src/main/java/com/example/base/ResultVO.java
  12. 35 0
      UtilsDemo-WX/src/main/java/com/example/base/RetHead.java
  13. 40 0
      UtilsDemo-WX/src/main/java/com/example/base/Token.java
  14. 42 0
      UtilsDemo-WX/src/main/java/com/example/config/FastJsonRedisSerializer.java
  15. 75 0
      UtilsDemo-WX/src/main/java/com/example/config/RedisConfig.java
  16. 49 0
      UtilsDemo-WX/src/main/java/com/example/config/Swagger2.java
  17. 41 0
      UtilsDemo-WX/src/main/java/com/example/config/WxMaConfiguration.java
  18. 24 0
      UtilsDemo-WX/src/main/java/com/example/config/WxMaProperties.java
  19. 42 0
      UtilsDemo-WX/src/main/java/com/example/config/WxMpConfiguration.java
  20. 34 0
      UtilsDemo-WX/src/main/java/com/example/config/WxMpProperties.java
  21. 49 0
      UtilsDemo-WX/src/main/java/com/example/config/WxPayConfiguration.java
  22. 97 0
      UtilsDemo-WX/src/main/java/com/example/config/WxPayProperties.java
  23. 129 0
      UtilsDemo-WX/src/main/java/com/example/constant/Constants.java
  24. 99 0
      UtilsDemo-WX/src/main/java/com/example/constant/GlobleConstant.java
  25. 54 0
      UtilsDemo-WX/src/main/java/com/example/controller/OrderController.java
  26. 146 0
      UtilsDemo-WX/src/main/java/com/example/controller/WechatAppController.java
  27. 233 0
      UtilsDemo-WX/src/main/java/com/example/controller/WechatAppPayController.java
  28. 205 0
      UtilsDemo-WX/src/main/java/com/example/controller/WechatH5Controller.java
  29. 337 0
      UtilsDemo-WX/src/main/java/com/example/controller/WechatH5PayController.java
  30. 29 0
      UtilsDemo-WX/src/main/java/com/example/entity/LoginRequest.java
  31. 184 0
      UtilsDemo-WX/src/main/java/com/example/entity/LoginUser.java
  32. 12 0
      UtilsDemo-WX/src/main/java/com/example/entity/MyWxMpUser.java
  33. 46 0
      UtilsDemo-WX/src/main/java/com/example/entity/OmsOrder.java
  34. 20 0
      UtilsDemo-WX/src/main/java/com/example/entity/RefundOrder.java
  35. 29 0
      UtilsDemo-WX/src/main/java/com/example/entity/User.java
  36. 19 0
      UtilsDemo-WX/src/main/java/com/example/entity/WxPayFrom.java
  37. 16 0
      UtilsDemo-WX/src/main/java/com/example/enums/DelEnum.java
  38. 29 0
      UtilsDemo-WX/src/main/java/com/example/enums/PayStatusEnum.java
  39. 142 0
      UtilsDemo-WX/src/main/java/com/example/enums/ResultEnum.java
  40. 16 0
      UtilsDemo-WX/src/main/java/com/example/enums/SexEnum.java
  41. 21 0
      UtilsDemo-WX/src/main/java/com/example/enums/WX_TypeEnum.java
  42. 17 0
      UtilsDemo-WX/src/main/java/com/example/exception/ExceptionContoller.java
  43. 65 0
      UtilsDemo-WX/src/main/java/com/example/filter/AuthGlobalFilter.java
  44. 10 0
      UtilsDemo-WX/src/main/java/com/example/mapper/UserMapper.java
  45. 231 0
      UtilsDemo-WX/src/main/java/com/example/redis/RedisCache.java
  46. 14 0
      UtilsDemo-WX/src/main/java/com/example/service/UserService.java
  47. 53 0
      UtilsDemo-WX/src/main/java/com/example/service/impl/UserServiceImpl.java
  48. 84 0
      UtilsDemo-WX/src/main/java/com/example/util/BeanCopyUtil.java
  49. 155 0
      UtilsDemo-WX/src/main/java/com/example/util/EscapeUtil.java
  50. 129 0
      UtilsDemo-WX/src/main/java/com/example/util/GeneratorIdUtils.java
  51. 45 0
      UtilsDemo-WX/src/main/java/com/example/util/IdUtils.java
  52. 194 0
      UtilsDemo-WX/src/main/java/com/example/util/IpUtils.java
  53. 880 0
      UtilsDemo-WX/src/main/java/com/example/util/RedisUtil.java
  54. 454 0
      UtilsDemo-WX/src/main/java/com/example/util/StringUtils.java
  55. 112 0
      UtilsDemo-WX/src/main/java/com/example/util/TokenUtil.java
  56. 486 0
      UtilsDemo-WX/src/main/java/com/example/util/UUID.java
  57. 69 0
      UtilsDemo-WX/src/main/java/com/example/util/UUIDUtils.java
  58. 55 0
      UtilsDemo-WX/src/main/java/com/example/util/WxUtils.java
  59. 22 0
      UtilsDemo-WX/src/main/java/com/example/vo/UserUpdateVO.java
  60. BIN
      UtilsDemo-WX/src/main/resources/apiclient_cert.p12
  61. 58 0
      UtilsDemo-WX/src/main/resources/application.yml
  62. 4 0
      UtilsDemo-WX/src/main/resources/mapper/User.xml
  63. 44 0
      UtilsDemo.iml
  64. 11 10
      pom.xml
  65. 168 0
      target/site/css/maven-base.css
  66. 161 0
      target/site/css/maven-theme.css
  67. 26 0
      target/site/css/print.css
  68. 1 0
      target/site/css/site.css
  69. 903 0
      target/site/dependencies.html
  70. BIN
      target/site/images/close.gif
  71. BIN
      target/site/images/collapsed.gif
  72. BIN
      target/site/images/expanded.gif
  73. BIN
      target/site/images/external.png
  74. BIN
      target/site/images/icon_error_sml.gif
  75. BIN
      target/site/images/icon_info_sml.gif
  76. BIN
      target/site/images/icon_success_sml.gif
  77. BIN
      target/site/images/icon_warning_sml.gif
  78. BIN
      target/site/images/logos/build-by-maven-black.png
  79. BIN
      target/site/images/logos/build-by-maven-white.png
  80. BIN
      target/site/images/logos/maven-feather.png
  81. BIN
      target/site/images/newwindow.png

+ 103 - 0
UtilsDemo-Excel/UtilsDemo-Excel.iml

@@ -0,0 +1,103 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module org.jetbrains.idea.maven.project.MavenProjectsManager.isMavenModule="true" type="JAVA_MODULE" version="4">
+  <component name="FacetManager">
+    <facet type="Spring" name="Spring">
+      <configuration />
+    </facet>
+    <facet type="web" name="Web">
+      <configuration>
+        <webroots />
+      </configuration>
+    </facet>
+  </component>
+  <component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_8">
+    <output url="file://$MODULE_DIR$/target/classes" />
+    <output-test url="file://$MODULE_DIR$/target/test-classes" />
+    <content url="file://$MODULE_DIR$">
+      <sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
+      <sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
+      <excludeFolder url="file://$MODULE_DIR$/target" />
+    </content>
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+    <orderEntry type="library" name="Maven: org.projectlombok:lombok:1.18.18" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-miniapp:3.9.0" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-common:3.9.0" level="project" />
+    <orderEntry type="library" name="Maven: com.thoughtworks.xstream:xstream:1.4.11.1" level="project" />
+    <orderEntry type="library" name="Maven: xmlpull:xmlpull:1.1.3.1" level="project" />
+    <orderEntry type="library" name="Maven: xpp3:xpp3_min:1.1.4c" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpclient:4.5.13" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpcore:4.4.14" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpmime:4.5.13" level="project" />
+    <orderEntry type="library" name="Maven: commons-codec:commons-codec:1.14" level="project" />
+    <orderEntry type="library" name="Maven: commons-io:commons-io:2.5" level="project" />
+    <orderEntry type="library" name="Maven: org.dom4j:dom4j:2.1.3" level="project" />
+    <orderEntry type="library" name="Maven: org.bouncycastle:bcpkix-jdk15on:1.65" level="project" />
+    <orderEntry type="library" name="Maven: org.bouncycastle:bcprov-jdk15on:1.65" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-mp:3.9.0" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-pay:3.9.0" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:qrcode-utils:1.1" level="project" />
+    <orderEntry type="library" name="Maven: com.google.zxing:core:3.2.1" level="project" />
+    <orderEntry type="library" name="Maven: org.jodd:jodd-http:5.1.6" level="project" />
+    <orderEntry type="library" name="Maven: org.jodd:jodd-core:5.1.6" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.commons:commons-lang3:3.10" level="project" />
+    <orderEntry type="library" name="Maven: commons-beanutils:commons-beanutils:1.9.4" level="project" />
+    <orderEntry type="library" name="Maven: commons-logging:commons-logging:1.2" level="project" />
+    <orderEntry type="library" name="Maven: commons-collections:commons-collections:3.2.2" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-databind:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-annotations:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-core:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.google.code.gson:gson:2.8.6" level="project" />
+    <orderEntry type="library" name="Maven: joda-time:joda-time:2.10.6" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-swagger2:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.swagger:swagger-annotations:1.5.14" level="project" />
+    <orderEntry type="library" name="Maven: io.swagger:swagger-models:1.5.14" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-spi:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-core:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: net.bytebuddy:byte-buddy:1.10.20" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-schema:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-swagger-common:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-spring-web:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: org.reflections:reflections:0.9.11" level="project" />
+    <orderEntry type="library" name="Maven: org.javassist:javassist:3.21.0-GA" level="project" />
+    <orderEntry type="library" name="Maven: com.google.guava:guava:20.0" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml:classmate:1.5.1" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:slf4j-api:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.plugin:spring-plugin-core:1.2.0.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-beans:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-context:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-aop:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.plugin:spring-plugin-metadata:1.2.0.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.mapstruct:mapstruct:1.2.0.Final" level="project" />
+    <orderEntry type="library" name="Maven: com.alibaba:fastjson:1.2.6" level="project" />
+    <orderEntry type="library" name="Maven: com.github.pagehelper:pagehelper:5.1.4" level="project" />
+    <orderEntry type="library" name="Maven: com.github.jsqlparser:jsqlparser:1.0" level="project" />
+    <orderEntry type="library" name="Maven: net.sf.dozer:dozer:5.5.1" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jcl-over-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-web:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-autoconfigure:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-logging:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-classic:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-core:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-to-slf4j:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-api:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jul-to-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: jakarta.annotation:jakarta.annotation-api:1.3.5" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-core:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-jcl:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.yaml:snakeyaml:1.26" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-json:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jdk8:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.module:jackson-module-parameter-names:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-tomcat:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-core:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.glassfish:jakarta.el:3.0.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-websocket:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-web:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-webmvc:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-expression:5.2.13.RELEASE" level="project" />
+  </component>
+</module>

+ 48 - 0
UtilsDemo-Excel/pom.xml

@@ -7,6 +7,54 @@
   <groupId>org.example</groupId>
   <artifactId>UtilsDemo-Excel</artifactId>
   <version>1.0-SNAPSHOT</version>
+  <dependencies>
+    <dependency>
+      <groupId>org.projectlombok</groupId>
+      <artifactId>lombok</artifactId>
+    </dependency>
+      <dependency>
+          <groupId>com.github.binarywang</groupId>
+          <artifactId>weixin-java-miniapp</artifactId>
+          <version>3.9.0</version>
+          <scope>compile</scope>
+      </dependency>
+    <dependency>
+      <groupId>com.github.binarywang</groupId>
+      <artifactId>weixin-java-mp</artifactId>
+      <version>3.9.0</version>
+      <scope>compile</scope>
+    </dependency>
+    <dependency>
+      <groupId>com.github.binarywang</groupId>
+      <artifactId>weixin-java-pay</artifactId>
+      <version>3.9.0</version>
+      <scope>compile</scope>
+    </dependency>
+    <dependency>
+      <groupId>io.springfox</groupId>
+      <artifactId>springfox-swagger2</artifactId>
+      <version>2.8.0</version>
+      <scope>compile</scope>
+    </dependency>
+    <dependency>
+      <groupId>com.alibaba</groupId>
+      <artifactId>fastjson</artifactId>
+      <version>1.2.6</version>
+      <scope>compile</scope>
+    </dependency>
+    <dependency>
+      <groupId>com.github.pagehelper</groupId>
+      <artifactId>pagehelper</artifactId>
+      <version>5.1.4</version>
+      <scope>compile</scope>
+    </dependency>
+    <dependency>
+      <groupId>net.sf.dozer</groupId>
+      <artifactId>dozer</artifactId>
+      <version>5.5.1</version>
+      <scope>compile</scope>
+    </dependency>
+  </dependencies>
 
   <parent>
     <groupId>org.example</groupId>

+ 57 - 0
UtilsDemo-SecureEncryption/UtilsDemo-SecureEncryption.iml

@@ -0,0 +1,57 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module org.jetbrains.idea.maven.project.MavenProjectsManager.isMavenModule="true" type="JAVA_MODULE" version="4">
+  <component name="FacetManager">
+    <facet type="Spring" name="Spring">
+      <configuration />
+    </facet>
+    <facet type="web" name="Web">
+      <configuration>
+        <webroots />
+      </configuration>
+    </facet>
+  </component>
+  <component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_8">
+    <output url="file://$MODULE_DIR$/target/classes" />
+    <output-test url="file://$MODULE_DIR$/target/test-classes" />
+    <content url="file://$MODULE_DIR$">
+      <sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
+      <sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
+      <excludeFolder url="file://$MODULE_DIR$/target" />
+    </content>
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-autoconfigure:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-core:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-jcl:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-context:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: commons-codec:commons-codec:1.13" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-web:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-logging:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-classic:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-core:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:slf4j-api:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-to-slf4j:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-api:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jul-to-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: jakarta.annotation:jakarta.annotation-api:1.3.5" level="project" />
+    <orderEntry type="library" name="Maven: org.yaml:snakeyaml:1.26" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-json:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-databind:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-annotations:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-core:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jdk8:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.module:jackson-module-parameter-names:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-tomcat:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-core:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.glassfish:jakarta.el:3.0.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-websocket:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-web:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-beans:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-webmvc:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-aop:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-expression:5.2.13.RELEASE" level="project" />
+  </component>
+</module>

+ 2 - 0
UtilsDemo-WX/.gitignore

@@ -0,0 +1,2 @@
+/.idea/
+/target/

+ 140 - 0
UtilsDemo-WX/UtilsDemo-WX.iml

@@ -0,0 +1,140 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module org.jetbrains.idea.maven.project.MavenProjectsManager.isMavenModule="true" type="JAVA_MODULE" version="4">
+  <component name="FacetManager">
+    <facet type="Spring" name="Spring">
+      <configuration />
+    </facet>
+    <facet type="web" name="Web">
+      <configuration>
+        <webroots />
+      </configuration>
+    </facet>
+  </component>
+  <component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_8">
+    <output url="file://$MODULE_DIR$/target/classes" />
+    <output-test url="file://$MODULE_DIR$/target/test-classes" />
+    <content url="file://$MODULE_DIR$">
+      <sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
+      <sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
+      <excludeFolder url="file://$MODULE_DIR$/target" />
+    </content>
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+    <orderEntry type="library" name="Maven: org.projectlombok:lombok:1.18.18" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.commons:commons-lang3:3.10" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-mp:3.1.0" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-common:3.1.0" level="project" />
+    <orderEntry type="library" name="Maven: com.thoughtworks.xstream:xstream:1.4.9" level="project" />
+    <orderEntry type="library" name="Maven: xmlpull:xmlpull:1.1.3.1" level="project" />
+    <orderEntry type="library" name="Maven: xpp3:xpp3_min:1.1.4c" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpclient:4.5.13" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpcore:4.4.14" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.httpcomponents:httpmime:4.5.13" level="project" />
+    <orderEntry type="library" name="Maven: com.google.code.gson:gson:2.8.6" level="project" />
+    <orderEntry type="library" name="Maven: commons-codec:commons-codec:1.14" level="project" />
+    <orderEntry type="library" name="Maven: commons-io:commons-io:2.5" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-pay:3.1.0" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:qrcode-utils:1.1" level="project" />
+    <orderEntry type="library" name="Maven: com.google.zxing:core:3.2.1" level="project" />
+    <orderEntry type="library" name="Maven: org.jodd:jodd-http:3.7.1" level="project" />
+    <orderEntry type="library" name="Maven: org.jodd:jodd-core:3.7.1" level="project" />
+    <orderEntry type="library" name="Maven: org.jodd:jodd-upload:3.7.1" level="project" />
+    <orderEntry type="library" name="Maven: commons-beanutils:commons-beanutils:1.9.3" level="project" />
+    <orderEntry type="library" name="Maven: commons-logging:commons-logging:1.2" level="project" />
+    <orderEntry type="library" name="Maven: commons-collections:commons-collections:3.2.2" level="project" />
+    <orderEntry type="library" name="Maven: org.bouncycastle:bcpkix-jdk15on:1.59" level="project" />
+    <orderEntry type="library" name="Maven: org.bouncycastle:bcprov-jdk15on:1.59" level="project" />
+    <orderEntry type="library" name="Maven: com.github.binarywang:weixin-java-miniapp:3.1.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-swagger-ui:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-spring-web:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: org.reflections:reflections:0.9.11" level="project" />
+    <orderEntry type="library" name="Maven: org.javassist:javassist:3.21.0-GA" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-swagger2:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.swagger:swagger-annotations:1.5.14" level="project" />
+    <orderEntry type="library" name="Maven: io.swagger:swagger-models:1.5.14" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-annotations:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-spi:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-core:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: net.bytebuddy:byte-buddy:1.10.20" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-schema:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: io.springfox:springfox-swagger-common:2.8.0" level="project" />
+    <orderEntry type="library" name="Maven: com.google.guava:guava:20.0" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml:classmate:1.5.1" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:slf4j-api:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.plugin:spring-plugin-core:1.2.0.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-beans:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-context:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-aop:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.plugin:spring-plugin-metadata:1.2.0.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.mapstruct:mapstruct:1.2.0.Final" level="project" />
+    <orderEntry type="library" name="Maven: com.github.pagehelper:pagehelper-spring-boot-starter:1.2.10" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-logging:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-classic:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-core:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-to-slf4j:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-api:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jul-to-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: jakarta.annotation:jakarta.annotation-api:1.3.5" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-core:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-jcl:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.yaml:snakeyaml:1.26" level="project" />
+    <orderEntry type="library" name="Maven: org.mybatis.spring.boot:mybatis-spring-boot-starter:1.3.2" level="project" />
+    <orderEntry type="library" name="Maven: org.mybatis.spring.boot:mybatis-spring-boot-autoconfigure:1.3.2" level="project" />
+    <orderEntry type="library" name="Maven: org.mybatis:mybatis:3.4.6" level="project" />
+    <orderEntry type="library" name="Maven: org.mybatis:mybatis-spring:1.3.2" level="project" />
+    <orderEntry type="library" name="Maven: com.github.pagehelper:pagehelper-spring-boot-autoconfigure:1.2.10" level="project" />
+    <orderEntry type="library" name="Maven: com.github.pagehelper:pagehelper:5.1.8" level="project" />
+    <orderEntry type="library" name="Maven: com.github.jsqlparser:jsqlparser:1.2" level="project" />
+    <orderEntry type="library" name="Maven: com.alibaba:fastjson:1.2.73" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-data-redis:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.data:spring-data-redis:2.3.7.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.data:spring-data-keyvalue:2.3.7.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.data:spring-data-commons:2.3.7.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-tx:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-oxm:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-context-support:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: io.lettuce:lettuce-core:5.3.6.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-common:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-handler:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-resolver:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-buffer:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-codec:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.netty:netty-transport:4.1.59.Final" level="project" />
+    <orderEntry type="library" name="Maven: io.projectreactor:reactor-core:3.3.14.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.reactivestreams:reactive-streams:1.0.3" level="project" />
+    <orderEntry type="library" name="Maven: net.sf.dozer:dozer:5.5.1" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jcl-over-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: com.baomidou:mybatisplus-spring-boot-starter:1.0.5" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-autoconfigure:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-configuration-processor:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-jdbc:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.zaxxer:HikariCP:3.4.5" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-jdbc:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.baomidou:mybatis-plus:2.1.8" level="project" />
+    <orderEntry type="library" name="Maven: com.baomidou:mybatis-plus-support:2.1.8" level="project" />
+    <orderEntry type="library" name="Maven: com.baomidou:mybatis-plus-core:2.1.8" level="project" />
+    <orderEntry type="library" name="Maven: com.baomidou:mybatis-plus-generate:2.1.8" level="project" />
+    <orderEntry type="library" name="Maven: tk.mybatis:mapper:3.4.5" level="project" />
+    <orderEntry type="library" name="Maven: javax.persistence:persistence-api:1.0" level="project" />
+    <orderEntry type="library" name="Maven: mysql:mysql-connector-java:8.0.19" level="project" />
+    <orderEntry type="library" name="Maven: io.jsonwebtoken:jjwt:0.9.1" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-databind:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-core:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: eu.bitwalker:UserAgentUtils:1.21" level="project" />
+    <orderEntry type="library" name="Maven: jakarta.validation:jakarta.validation-api:2.0.2" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-web:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-json:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jdk8:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.module:jackson-module-parameter-names:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-tomcat:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-core:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.glassfish:jakarta.el:3.0.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-websocket:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-web:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-webmvc:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-expression:5.2.13.RELEASE" level="project" />
+  </component>
+</module>

+ 145 - 0
UtilsDemo-WX/pom.xml

@@ -0,0 +1,145 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
+    <modelVersion>4.0.0</modelVersion>
+
+    <groupId>org.example</groupId>
+    <artifactId>UtilsDemo-WX</artifactId>
+    <version>1.0-SNAPSHOT</version>
+
+    <parent>
+        <groupId>org.example</groupId>
+        <artifactId>UtilsDemo</artifactId>
+        <version>1.0-SNAPSHOT</version>
+    </parent>
+
+    <name>UtilsDemo-WX</name>
+
+
+    <properties>
+        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
+        <binarywang.weixin.version>3.1.0</binarywang.weixin.version>
+        <pagehelper-starter.version>1.2.10</pagehelper-starter.version>
+        <bitwalker.version>1.21</bitwalker.version>
+    </properties>
+
+    <dependencies>
+        <dependency>
+            <groupId>org.projectlombok</groupId>
+            <artifactId>lombok</artifactId>
+        </dependency>
+
+        <dependency>
+            <groupId>org.apache.commons</groupId>
+            <artifactId>commons-lang3</artifactId>
+            <version>3.10</version>
+        </dependency>
+
+
+        <!-- weixin-java-mp -->
+        <dependency>
+            <groupId>com.github.binarywang</groupId>
+            <artifactId>weixin-java-mp</artifactId>
+            <version>${binarywang.weixin.version}</version>
+        </dependency>
+
+        <!-- weixin-java-pay -->
+        <dependency>
+            <groupId>com.github.binarywang</groupId>
+            <artifactId>weixin-java-pay</artifactId>
+            <version>${binarywang.weixin.version}</version>
+        </dependency>
+
+        <!-- weixin-java-miniapp -->
+        <dependency>
+            <groupId>com.github.binarywang</groupId>
+            <artifactId>weixin-java-miniapp</artifactId>
+            <version>${binarywang.weixin.version}</version>
+        </dependency>
+
+        <dependency>
+            <groupId>io.springfox</groupId>
+            <artifactId>springfox-swagger-ui</artifactId>
+            <version>2.8.0</version>
+        </dependency>
+        <dependency>
+            <groupId>io.springfox</groupId>
+            <artifactId>springfox-swagger2</artifactId>
+            <version>2.8.0</version>
+        </dependency>
+
+        <!--MyBatis分页插件starter-->
+        <dependency>
+            <groupId>com.github.pagehelper</groupId>
+            <artifactId>pagehelper-spring-boot-starter</artifactId>
+            <version>${pagehelper-starter.version}</version>
+        </dependency>
+
+        <dependency>
+            <groupId>com.alibaba</groupId>
+            <artifactId>fastjson</artifactId>
+            <version>1.2.73</version>
+        </dependency>
+
+
+        <dependency>
+            <groupId>org.springframework.boot</groupId>
+            <artifactId>spring-boot-starter-data-redis</artifactId>
+        </dependency>
+
+        <dependency>
+            <groupId>net.sf.dozer</groupId>
+            <artifactId>dozer</artifactId>
+            <version>5.5.1</version>
+        </dependency>
+
+        <dependency>
+            <groupId>com.baomidou</groupId>
+            <artifactId>mybatisplus-spring-boot-starter</artifactId>
+            <version>1.0.5</version>
+        </dependency>
+        <dependency>
+            <groupId>com.baomidou</groupId>
+            <artifactId>mybatis-plus</artifactId>
+            <version>2.1.8</version>
+        </dependency>
+        <dependency>
+            <groupId>tk.mybatis</groupId>
+            <artifactId>mapper</artifactId>
+            <version>3.4.5</version>
+        </dependency>
+
+        <dependency>
+            <groupId>mysql</groupId>
+            <artifactId>mysql-connector-java</artifactId>
+            <version>8.0.19</version>
+        </dependency>
+
+<!--        <dependency>-->
+<!--            <groupId>org.springframework.boot</groupId>-->
+<!--            <artifactId>spring-boot-starter-security</artifactId>-->
+<!--        </dependency>-->
+
+        <!--Token生成与解析-->
+        <dependency>
+            <groupId>io.jsonwebtoken</groupId>
+            <artifactId>jjwt</artifactId>
+            <version>0.9.1</version>
+        </dependency>
+
+        <!-- 解析客户端操作系统、浏览器等 -->
+        <dependency>
+            <groupId>eu.bitwalker</groupId>
+            <artifactId>UserAgentUtils</artifactId>
+            <version>${bitwalker.version}</version>
+        </dependency>
+        <dependency>
+            <groupId>jakarta.validation</groupId>
+            <artifactId>jakarta.validation-api</artifactId>
+            <version>2.0.2</version>
+        </dependency>
+
+    </dependencies>
+
+</project>

+ 19 - 0
UtilsDemo-WX/src/main/java/com/example/WXApplication.java

@@ -0,0 +1,19 @@
+package com.example;
+
+import org.mybatis.spring.annotation.MapperScan;
+import org.springframework.boot.SpringApplication;
+import org.springframework.boot.autoconfigure.SpringBootApplication;
+import org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration;
+import org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration;
+import springfox.documentation.swagger2.annotations.EnableSwagger2;
+
+
+@SpringBootApplication
+@EnableSwagger2
+@MapperScan("com.example.mapper")
+public class WXApplication {
+
+    public static void main(String[] args) {
+        SpringApplication.run(WXApplication.class, args);
+    }
+}

+ 185 - 0
UtilsDemo-WX/src/main/java/com/example/base/BaseController.java

@@ -0,0 +1,185 @@
+package com.example.base;
+
+import com.example.enums.ResultEnum;
+import com.example.util.BeanCopyUtil;
+import org.springframework.stereotype.Component;
+import org.springframework.web.context.request.RequestContextHolder;
+import org.springframework.web.context.request.ServletRequestAttributes;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.util.Collection;
+
+/**
+ * 控制器基类
+ *
+ * @author yys
+ */
+@Component
+@SuppressWarnings({"rawtypes", "unchecked"})
+public class BaseController {
+
+    protected static final String ACCESS_TOKEN = "accessToken";
+
+    /**
+     * 无业务数据成功返回
+     *
+     * @return
+     */
+    protected ResultVO success() {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setData(null);
+        return resultVO;
+    }
+
+    /**
+     * 无业务数据失败返回
+     *
+     * @return
+     */
+    protected ResultVO failure() {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setCode(ResultEnum.FAILURE.getCode());
+        resultVO.setMsg(ResultEnum.FAILURE.getMsg());
+        resultVO.setData(null);
+        return resultVO;
+    }
+
+    protected ResultVO failure(String message) {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setCode(ResultEnum.FAILURE.getCode());
+        resultVO.setMsg(message);
+        resultVO.setData(null);
+        return resultVO;
+    }
+
+    /**
+     * 有业务数据成功返回
+     *
+     * @param data
+     * @return
+     */
+    protected ResultVO success(Object data, long count) {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setData(data);
+        resultVO.setCount(count);
+        return resultVO;
+    }
+
+    /**
+     * 有业务数据成功返回
+     *
+     * @param data
+     * @return
+     */
+    protected ResultVO success(Object data) {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setData(data);
+        return resultVO;
+    }
+
+    /**
+     * 有业务数据成功返回
+     *
+     * @param data
+     * @return
+     */
+    protected ResultVO error(String msg) {
+        ResultVO resultVO = new ResultVO();
+        resultVO.setCode(1);
+        resultVO.setData(msg);
+        return resultVO;
+    }
+
+    /**
+     * 操作成功后,返回结果(前后端分离)
+     *
+     * @param resultVO
+     * @return
+     */
+    protected ResponseBase responseSuccess(ResultVO resultVO) {
+        ResponseBase responseBase = new ResponseBase(resultVO.getData());
+        RetHead retHead = new RetHead();
+        retHead.setErrCode(resultVO.getCode().shortValue());
+        retHead.setErrMsg(resultVO.getMsg());
+        retHead.setTotal(resultVO.getCount());
+        responseBase.setRetHead(retHead);
+        return responseBase;
+    }
+
+
+    /**
+     * 操作异常后,返回结果(前后端分离)
+     *
+     * @param resultVO
+     * @return
+     */
+    protected ResponseBase responseError(ResultVO resultVO) {
+        RetHead retHead = new RetHead();
+        retHead.setErrCode(resultVO.getCode().shortValue());
+        retHead.setErrMsg(resultVO.getMsg());
+        return new ResponseBase(retHead);
+    }
+
+    /**
+     * 操作异常后,返回结果(前后端分离)
+     *
+     * @return
+     */
+    protected ResponseBase responseError(Integer errCode, String errMsg) {
+        RetHead retHead = new RetHead();
+        retHead.setErrCode(errCode.shortValue());
+        retHead.setErrMsg(errMsg);
+        return new ResponseBase(retHead);
+    }
+
+    /**
+     * 成功返回并转换data对象类型为目标对象类型
+     *
+     * @param data
+     * @param destinationClass 目标Class对象
+     * @return
+     */
+    protected ResultVO success(Object data, Class destinationClass) {
+        ResultVO resultVO = new ResultVO();
+        if (data != null && data instanceof Collection) {
+            Collection collection = (Collection) data;
+            resultVO.setData(BeanCopyUtil.convertList(collection, destinationClass));
+        } else {
+            resultVO.setData(data);
+        }
+        return resultVO;
+    }
+
+    /**
+     * 判断调用服务是否成功
+     *
+     * @param resultVO
+     * @return
+     */
+    public boolean isSuccess(ResultVO resultVO) {
+        return ResultEnum.SUCCESS.getCode().equals(resultVO.getCode());
+    }
+
+    /**
+     * 获取response对象
+     *
+     * @return
+     */
+    protected HttpServletResponse getResponseObject() {
+        ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder
+                .getRequestAttributes();
+        return requestAttributes.getResponse();
+    }
+
+    /**
+     * 获取request对象
+     *
+     * @return
+     */
+    protected HttpServletRequest getRequestObject() {
+        ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder
+                .getRequestAttributes();
+        return requestAttributes.getRequest();
+    }
+}

+ 9 - 0
UtilsDemo-WX/src/main/java/com/example/base/BaseEntity.java

@@ -0,0 +1,9 @@
+package com.example.base;
+
+import java.io.Serializable;
+
+/**
+ * @author xiezt
+ */
+public class BaseEntity implements Serializable {
+}

+ 32 - 0
UtilsDemo-WX/src/main/java/com/example/base/ResponseBase.java

@@ -0,0 +1,32 @@
+package com.example.base;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+/**
+ * @author xiezt
+ */
+@Data
+@ApiModel(value = "web响应类", description = "web响应类")
+public class ResponseBase<T> extends BaseEntity {
+
+    @ApiModelProperty(value = "返回头信息")
+    private RetHead retHead;
+
+    @ApiModelProperty(value = "返回内容信息")
+    private T retBody;
+
+    public ResponseBase() {
+
+    }
+
+    public ResponseBase(T retBody) {
+        this.retBody = retBody;
+    }
+
+    public ResponseBase(RetHead retHead) {
+        this.retHead = retHead;
+    }
+
+}

+ 87 - 0
UtilsDemo-WX/src/main/java/com/example/base/ResultVO.java

@@ -0,0 +1,87 @@
+package com.example.base;
+
+import com.example.enums.ResultEnum;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+/**
+ * 服务调用统一返回封装类
+ *
+ * @param <T>
+ * @author yys
+ */
+@Data
+@ApiModel(value = "client响应类", description = "client响应类")
+public class ResultVO<T> {
+
+    private static final Integer DEFAULT_COUNT = 0;
+    /**
+     * 状态码 (默认值:0)
+     */
+    @ApiModelProperty(value = "响应码", example = "响应码")
+    private Integer code = ResultEnum.SUCCESS.getCode();
+    /**
+     * 状态提示信息 (默认值:成功)
+     */
+    @ApiModelProperty(value = "返回信息", example = "返回信息")
+    private String msg = ResultEnum.SUCCESS.getMsg();
+
+    /**
+     * 记录总数,默认是0
+     */
+    private long count = DEFAULT_COUNT;
+    /**
+     * 业务数据
+     */
+    @ApiModelProperty(value = "返回内容", example = "返回内容")
+    private T data;
+
+    public ResultVO() {
+        super();
+    }
+
+    public ResultVO(Integer code, String msg) {
+        super();
+        this.code = code;
+        this.msg = msg;
+    }
+
+    public ResultVO(Integer code, String msg, T data) {
+        super();
+        this.code = code;
+        this.msg = msg;
+        this.data = data;
+    }
+
+    public ResultVO(Integer code, String msg, long count, T data) {
+        super();
+        this.code = code;
+        this.msg = msg;
+        this.count = count;
+        this.data = data;
+    }
+
+    public ResultVO(ResultEnum resultEnum) {
+        super();
+        this.code = resultEnum.getCode();
+        this.msg = resultEnum.getMsg();
+    }
+
+    public ResultVO(ResultEnum resultEnum, T data, long count) {
+        super();
+        this.code = resultEnum.getCode();
+        this.msg = resultEnum.getMsg();
+        this.data = data;
+        this.count = count;
+    }
+
+    public ResultVO(ResultEnum resultEnum, T data) {
+        super();
+        this.code = resultEnum.getCode();
+        this.msg = resultEnum.getMsg();
+        this.data = data;
+        this.count = count;
+    }
+
+}

+ 35 - 0
UtilsDemo-WX/src/main/java/com/example/base/RetHead.java

@@ -0,0 +1,35 @@
+package com.example.base;
+
+import com.example.enums.ResultEnum;
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+
+/**
+ * @author xiezt
+ */
+@Data
+@ApiModel(value = "接口返回信息", description = "接口返回信息")
+public class RetHead extends BaseEntity {
+
+    public RetHead() {
+
+    }
+
+    public RetHead(ResultEnum resultEnum) {
+        this.errCode = Short.valueOf(resultEnum.getCode().toString());
+        this.errMsg = resultEnum.getMsg();
+    }
+
+    @ApiModelProperty(value = "返回响应码", example = "返回响应码")
+    private short errCode;
+
+    @ApiModelProperty(value = "返回响应信息", example = "返回响应信息")
+    private String errMsg;
+
+    @ApiModelProperty(value = "总记录数")
+    private Long total = 0L;
+
+
+}

+ 40 - 0
UtilsDemo-WX/src/main/java/com/example/base/Token.java

@@ -0,0 +1,40 @@
+package com.example.base;
+
+import lombok.Data;
+
+import java.sql.Timestamp;
+
+/**
+ * Token属性定义
+ *
+ * @author yys
+ */
+@Data
+public class Token {
+
+    /**
+     * Token令牌
+     */
+    private String accessToken;
+    /**
+     * 过期时间
+     */
+    private Timestamp expireTime;
+
+    /**
+     * 刷新access_token
+     */
+    private String refreshToken;
+
+    public Token() {
+        super();
+    }
+
+    public Token(String accessToken, Timestamp expireTime) {
+        super();
+        this.accessToken = accessToken;
+        this.expireTime = expireTime;
+    }
+
+
+}

+ 42 - 0
UtilsDemo-WX/src/main/java/com/example/config/FastJsonRedisSerializer.java

@@ -0,0 +1,42 @@
+package com.example.config;
+
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.serializer.SerializerFeature;
+import org.springframework.data.redis.serializer.RedisSerializer;
+import org.springframework.data.redis.serializer.SerializationException;
+
+import java.nio.charset.Charset;
+
+/**
+ * @author: Oliver.li
+ * @Description: 重写FastJsonRedisSerializer,能够反序列化为正确的Java类型
+ * @date: 2018/7/5 16:10
+ */
+public class FastJsonRedisSerializer<T> implements RedisSerializer<T> {
+
+    public static final Charset DEFAULT_CHARSET = Charset.forName("UTF-8");
+
+    private Class<T> clazz;
+
+    public FastJsonRedisSerializer(Class<T> clazz) {
+        super();
+        this.clazz = clazz;
+    }
+
+    @Override
+    public byte[] serialize(T t) throws SerializationException {
+        if (t == null) {
+            return new byte[0];
+        }
+        return JSON.toJSONString(t, SerializerFeature.WriteClassName).getBytes(DEFAULT_CHARSET);
+    }
+
+    @Override
+    public T deserialize(byte[] bytes) throws SerializationException {
+        if (bytes == null || bytes.length <= 0) {
+            return null;
+        }
+        String str = new String(bytes, DEFAULT_CHARSET);
+        return JSON.parseObject(str, clazz);
+    }
+}

+ 75 - 0
UtilsDemo-WX/src/main/java/com/example/config/RedisConfig.java

@@ -0,0 +1,75 @@
+package com.example.config;
+
+import com.alibaba.fastjson.parser.ParserConfig;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.core.io.ClassPathResource;
+import org.springframework.data.redis.cache.RedisCacheConfiguration;
+import org.springframework.data.redis.connection.RedisConnectionFactory;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.data.redis.core.script.DefaultRedisScript;
+import org.springframework.data.redis.serializer.RedisSerializationContext;
+import org.springframework.data.redis.serializer.StringRedisSerializer;
+import org.springframework.scripting.support.ResourceScriptSource;
+
+import java.time.Duration;
+
+/**
+ * @author: Oliver.li
+ * @Description:
+ * @date: 2018/6/26 10:21
+ */
+@Configuration
+public class RedisConfig {
+
+    /**
+     * 设置redisTemplate序列化方式
+     * @param redisConnectionFactory
+     * @return
+     */
+    @Bean
+    public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory redisConnectionFactory) {
+        com.alibaba.fastjson.support.spring.FastJsonRedisSerializer<Object> fastJsonRedisSerializer =
+                new com.alibaba.fastjson.support.spring.FastJsonRedisSerializer<>(Object.class);
+
+        StringRedisSerializer stringRedisSerializer = new StringRedisSerializer();
+
+        RedisTemplate<String, Object> template = new RedisTemplate<>();
+        template.setConnectionFactory(redisConnectionFactory);
+        template.setKeySerializer(stringRedisSerializer);
+        template.setHashKeySerializer(stringRedisSerializer);
+        template.setValueSerializer(fastJsonRedisSerializer);
+        template.setHashValueSerializer(fastJsonRedisSerializer);
+        template.afterPropertiesSet();
+        return template;
+    }
+
+    /**
+     * 设置 @Cacheable 序列化方式
+     * 设置过期时间为1天
+     * @return
+     */
+    @Bean
+    public RedisCacheConfiguration redisCacheConfiguration(){
+        FastJsonRedisSerializer<Object> serializer = new FastJsonRedisSerializer<>(Object.class);
+        ParserConfig parserConfig = ParserConfig.getGlobalInstance();
+        parserConfig.addAccept("com.llf.demo.");
+        parserConfig.addAccept("com.github.pagehelper.");
+
+        RedisCacheConfiguration configuration = RedisCacheConfiguration.defaultCacheConfig();
+        configuration = configuration.serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(serializer)).entryTtl(Duration.ofDays(1));
+        return configuration;
+    }
+
+    /**
+     * 分布式限流lua脚本
+     * @return
+     */
+    @Bean
+    public DefaultRedisScript<Number> rateLimitLua(){
+        DefaultRedisScript<Number> redisScript = new DefaultRedisScript<>();
+        redisScript.setScriptSource(new ResourceScriptSource(new ClassPathResource("rateLimit.lua")));
+        redisScript.setResultType(Number.class);
+        return redisScript;
+    }
+}

+ 49 - 0
UtilsDemo-WX/src/main/java/com/example/config/Swagger2.java

@@ -0,0 +1,49 @@
+package com.example.config;
+
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import springfox.documentation.builders.ApiInfoBuilder;
+import springfox.documentation.builders.PathSelectors;
+import springfox.documentation.builders.RequestHandlerSelectors;
+import springfox.documentation.service.ApiInfo;
+import springfox.documentation.spi.DocumentationType;
+import springfox.documentation.spring.web.plugins.Docket;
+import springfox.documentation.swagger2.annotations.EnableSwagger2;
+
+@Configuration
+@EnableSwagger2
+public class Swagger2 {
+    /**
+     * 创建API应用
+     * apiInfo() 增加API相关信息
+     * 通过select()函数返回一个ApiSelectorBuilder实例,用来控制哪些接口暴露给Swagger来展现,
+     * 本例采用指定扫描的包路径来定义指定要建立API的目录。
+     *
+     * @return
+     */
+    @Bean
+    public Docket createRestApi() {
+        return new Docket(DocumentationType.SWAGGER_2)
+                .apiInfo(apiInfo())
+                .select()
+                .apis(RequestHandlerSelectors.basePackage("com.example.controller"))
+                .paths(PathSelectors.any())
+                .build();
+    }
+
+    /**
+     * 创建该API的基本信息(这些基本信息会展现在文档页面中)
+     * 访问地址:http://项目实际地址/swagger-ui.html
+     *
+     * @return
+     */
+    private ApiInfo apiInfo() {
+        return new ApiInfoBuilder()
+                .title("Spring Boot中使用Swagger2构建RESTful APIs")
+                .description("更多请关注http://www.baidu.com")
+                .termsOfServiceUrl("http://www.baidu.com")
+                .contact("sunf")
+                .version("1.0")
+                .build();
+    }
+}

+ 41 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxMaConfiguration.java

@@ -0,0 +1,41 @@
+package com.example.config;
+
+import cn.binarywang.wx.miniapp.api.WxMaService;
+import cn.binarywang.wx.miniapp.api.impl.WxMaServiceImpl;
+import cn.binarywang.wx.miniapp.config.WxMaInMemoryConfig;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+@Configuration
+@ConditionalOnClass(WxMaService.class)
+public class WxMaConfiguration {
+
+    /**
+     * 设置微信小程序的appid
+     */
+    @Value("${wx.ma.appId}")
+    private String appId;
+
+    /**
+     * 设置微信小程序的app secret
+     */
+    @Value("${wx.ma.secret}")
+
+    private String secret;
+
+
+    @Bean
+    @ConditionalOnMissingBean
+    public WxMaService wxMaService() {
+        WxMaInMemoryConfig config = new WxMaInMemoryConfig();
+        config.setAppid(appId);
+        config.setSecret(secret);
+        WxMaService wxMaService = new WxMaServiceImpl();
+        wxMaService.setWxMaConfig(config);
+        return wxMaService;
+    }
+
+}

+ 24 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxMaProperties.java

@@ -0,0 +1,24 @@
+package com.example.config;
+
+
+import lombok.Data;
+import org.springframework.boot.context.properties.ConfigurationProperties;
+
+
+/**
+ * @author xiezt
+ */
+@Data
+@ConfigurationProperties(prefix = "wx.ma")
+public class WxMaProperties {
+    /**
+     * 设置微信公众号的appid
+     */
+    private String appId;
+
+    /**
+     * 设置微信公众号的app secret
+     */
+    private String secret;
+
+}

+ 42 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxMpConfiguration.java

@@ -0,0 +1,42 @@
+package com.example.config;
+
+
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.mp.api.WxMpInMemoryConfigStorage;
+import me.chanjar.weixin.mp.api.WxMpService;
+import me.chanjar.weixin.mp.api.impl.WxMpServiceImpl;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
+import org.springframework.boot.context.properties.EnableConfigurationProperties;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+@Slf4j
+@Configuration
+@ConditionalOnClass(WxMpService.class)
+@EnableConfigurationProperties(WxMpProperties.class)
+public class WxMpConfiguration {
+
+    private WxMpProperties properties;
+
+    @Autowired
+    public WxMpConfiguration(WxMpProperties properties) {
+        this.properties = properties;
+    }
+
+    @Bean
+    @ConditionalOnMissingBean
+    public WxMpService wxMpService() {
+        WxMpInMemoryConfigStorage configStorage = new WxMpInMemoryConfigStorage();
+        configStorage.setAppId(StringUtils.trimToNull(this.properties.getAppId()));
+        configStorage.setSecret(StringUtils.trimToNull(this.properties.getSecret()));
+        configStorage.setToken(StringUtils.trimToNull(this.properties.getToken()));
+        configStorage.setAesKey(StringUtils.trimToNull(this.properties.getAesKey()));
+        WxMpService wxMpService = new WxMpServiceImpl();
+        wxMpService.setWxMpConfigStorage(configStorage);
+        return wxMpService;
+    }
+}
+

+ 34 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxMpProperties.java

@@ -0,0 +1,34 @@
+package com.example.config;
+
+
+import lombok.Data;
+import org.springframework.boot.context.properties.ConfigurationProperties;
+
+
+/**
+ * @author xiezt
+ */
+@Data
+@ConfigurationProperties(prefix = "wx.mp")
+public class WxMpProperties {
+    /**
+     * 设置微信公众号的appid
+     */
+    private String appId;
+
+    /**
+     * 设置微信公众号的app secret
+     */
+    private String secret;
+
+    /**
+     * 设置微信公众号的token
+     */
+    private String token;
+
+    /**
+     * 设置微信公众号的EncodingAESKey
+     */
+    private String aesKey;
+
+}

+ 49 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxPayConfiguration.java

@@ -0,0 +1,49 @@
+package com.example.config;
+
+import com.github.binarywang.wxpay.config.WxPayConfig;
+import com.github.binarywang.wxpay.service.WxPayService;
+import com.github.binarywang.wxpay.service.impl.WxPayServiceImpl;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
+import org.springframework.boot.context.properties.EnableConfigurationProperties;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+
+/**
+ * @author Binary Wang
+ */
+@Configuration
+@ConditionalOnClass(WxPayService.class)
+@EnableConfigurationProperties(WxPayProperties.class)
+public class WxPayConfiguration {
+
+  private WxPayProperties properties;
+
+  @Autowired
+  public WxPayConfiguration(WxPayProperties properties) {
+    this.properties = properties;
+  }
+
+  @Bean
+  @ConditionalOnMissingBean
+  public WxPayService wxService() {
+    WxPayConfig payConfig = new WxPayConfig();
+    payConfig.setAppId(StringUtils.trimToNull(this.properties.getAppId()));
+    payConfig.setMchId(StringUtils.trimToNull(this.properties.getMchId()));
+    payConfig.setMchKey(StringUtils.trimToNull(this.properties.getMchKey()));
+    payConfig.setSubAppId(StringUtils.trimToNull(this.properties.getSubAppId()));
+    payConfig.setSubMchId(StringUtils.trimToNull(this.properties.getSubMchId()));
+    payConfig.setKeyPath(StringUtils.trimToNull(this.properties.getKeyPath()));
+
+
+    // 可以指定是否使用沙箱环境
+    payConfig.setUseSandboxEnv(false);
+
+    WxPayService wxPayService = new WxPayServiceImpl();
+    wxPayService.setConfig(payConfig);
+    return wxPayService;
+  }
+
+}

+ 97 - 0
UtilsDemo-WX/src/main/java/com/example/config/WxPayProperties.java

@@ -0,0 +1,97 @@
+package com.example.config;
+
+import org.apache.commons.lang3.builder.ToStringBuilder;
+import org.apache.commons.lang3.builder.ToStringStyle;
+import org.springframework.boot.context.properties.ConfigurationProperties;
+
+/**
+ * wxpay pay properties
+ *
+ * @author Binary Wang
+ */
+@ConfigurationProperties(prefix = "wx.pay")
+public class WxPayProperties {
+  /**
+   * 设置微信公众号或者小程序等的appid
+   */
+  private String appId;
+
+  /**
+   * 微信支付商户号
+   */
+  private String mchId;
+
+  /**
+   * 微信支付商户密钥
+   */
+  private String mchKey;
+
+  /**
+   * 服务商模式下的子商户公众账号ID,普通模式请不要配置,请在配置文件中将对应项删除
+   */
+  private String subAppId;
+
+  /**
+   * 服务商模式下的子商户号,普通模式请不要配置,最好是请在配置文件中将对应项删除
+   */
+  private String subMchId;
+
+  /**
+   * apiclient_cert.p12文件的绝对路径,或者如果放在项目中,请以classpath:开头指定
+   */
+  private String keyPath;
+
+  public String getAppId() {
+    return this.appId;
+  }
+
+  public void setAppId(String appId) {
+    this.appId = appId;
+  }
+
+  public String getMchId() {
+    return mchId;
+  }
+
+  public void setMchId(String mchId) {
+    this.mchId = mchId;
+  }
+
+  public String getMchKey() {
+    return mchKey;
+  }
+
+  public void setMchKey(String mchKey) {
+    this.mchKey = mchKey;
+  }
+
+  public String getSubAppId() {
+    return subAppId;
+  }
+
+  public void setSubAppId(String subAppId) {
+    this.subAppId = subAppId;
+  }
+
+  public String getSubMchId() {
+    return subMchId;
+  }
+
+  public void setSubMchId(String subMchId) {
+    this.subMchId = subMchId;
+  }
+
+  public String getKeyPath() {
+    return this.keyPath;
+  }
+
+  public void setKeyPath(String keyPath) {
+    this.keyPath = keyPath;
+  }
+
+  @Override
+  public String toString() {
+    return ToStringBuilder.reflectionToString(this,
+        ToStringStyle.MULTI_LINE_STYLE);
+  }
+}

+ 129 - 0
UtilsDemo-WX/src/main/java/com/example/constant/Constants.java

@@ -0,0 +1,129 @@
+package com.example.constant;
+
+/**
+ * 通用常量信息
+ * 
+ * @author ruoyi
+ */
+public class Constants
+{
+    /**
+     * UTF-8 字符集
+     */
+    public static final String UTF8 = "UTF-8";
+
+    /**
+     * GBK 字符集
+     */
+    public static final String GBK = "GBK";
+
+    /**
+     * http请求
+     */
+    public static final String HTTP = "http://";
+
+    /**
+     * https请求
+     */
+    public static final String HTTPS = "https://";
+
+    /**
+     * 通用成功标识
+     */
+    public static final String SUCCESS = "0";
+
+    /**
+     * 通用失败标识
+     */
+    public static final String FAIL = "1";
+
+    /**
+     * 登录成功
+     */
+    public static final String LOGIN_SUCCESS = "Success";
+
+    /**
+     * 注销
+     */
+    public static final String LOGOUT = "Logout";
+
+    /**
+     * 登录失败
+     */
+    public static final String LOGIN_FAIL = "Error";
+
+    /**
+     * 验证码 redis key
+     */
+    public static final String CAPTCHA_CODE_KEY = "captcha_codes:";
+
+    /**
+     * 登录用户 redis key
+     */
+    public static final String LOGIN_TOKEN_KEY = "login_tokens:";
+    
+    /**
+     * 防重提交 redis key
+     */
+    public static final String REPEAT_SUBMIT_KEY = "repeat_submit:";
+
+    /**
+     * 验证码有效期(分钟)
+     */
+    public static final Integer CAPTCHA_EXPIRATION = 2;
+
+    /**
+     * 令牌
+     */
+    public static final String TOKEN = "token";
+
+    /**
+     * 令牌前缀
+     */
+    public static final String TOKEN_PREFIX = "Bearer ";
+
+    /**
+     * 令牌前缀
+     */
+    public static final String LOGIN_USER_KEY = "login_user_key";
+
+    /**
+     * 用户ID
+     */
+    public static final String JWT_USERID = "userid";
+
+    /**
+     * 用户名称
+     */
+    public static final String JWT_USERNAME = "sub";
+
+    /**
+     * 用户头像
+     */
+    public static final String JWT_AVATAR = "avatar";
+
+    /**
+     * 创建时间
+     */
+    public static final String JWT_CREATED = "created";
+
+    /**
+     * 用户权限
+     */
+    public static final String JWT_AUTHORITIES = "authorities";
+
+    /**
+     * 参数管理 cache key
+     */
+    public static final String SYS_CONFIG_KEY = "sys_config:";
+
+    /**
+     * 字典管理 cache key
+     */
+    public static final String SYS_DICT_KEY = "sys_dict:";
+
+    /**
+     * 资源映射路径 前缀
+     */
+    public static final String RESOURCE_PREFIX = "/profile";
+}

+ 99 - 0
UtilsDemo-WX/src/main/java/com/example/constant/GlobleConstant.java

@@ -0,0 +1,99 @@
+package com.example.constant;
+
+/**
+ * 全局参数定义
+ *
+ * @author yys
+ */
+public class GlobleConstant {
+    /**
+     * 前台登录用户缓存参数
+     */
+    public final static String CURRENT_MEMBER = "CurrentMember";
+    /**
+     * 后台登录用户缓存参数
+     */
+    public final static String CURRENT_USER = "CurrentUser";
+    /**
+     * 前台传入accessToken属性名称
+     */
+    public final static String ACCESS_TOKEN = "accessToken";
+    /**
+     * 默认feign调用token标识
+     */
+    public final static String FEIGN_TOKEN = "feignToken";
+    /**
+     * 机构主键标识
+     */
+    public final static String ORG_GUID = "orgGuid";
+    /**
+     * 用户主键标识
+     */
+    public final static String USER_GUID = "userGuid";
+    /**
+     * 机构对象标识
+     */
+    public final static String ORG_OBJECT = "orgObject";
+
+    /**
+     * 店铺信息
+     */
+    public final static String ORG_STORE_OBJ = "storeObject";
+    /**
+     * 用户权限列表缓存参数
+     */
+    public final static String AUTH_LIST = "authList";
+    /**
+     * 用户权限map缓存参数
+     */
+    public final static String AUTH_MAP = "authMap";
+
+    /**
+     * 用户菜单列表缓存
+     */
+    public final static String AUTH_MENN_OBJ = "authMenuObj";
+    /**
+     * accesstoken 缓存时间 单位秒
+     */
+    public final static long TOKEN_EXPIRES_SECOND =  6 * 3600;
+
+    /**
+     * 访问Token过期时间
+     */
+    public final static long ACCESS_TOKEN_EXPIRES_SECOND = 30 * 60;
+
+    /**
+     * refreshtoken 过期时间
+     */
+    public final static long REFRESH_TOKEN_EXPIRES_SECOND = 2 * 7 * 24 * 3600;
+    /**
+     * accesstoken aud标识
+     */
+    public final static String TOKEN_AUD = "tokenAud";
+    /**
+     * refreshtoken aud标识
+     */
+    public final static String REFRESH_TOKEN_AUD = "refreshTokenAud";
+
+    public final static String UPLOAD_RELATIVE_PATH = "/images/";
+
+    public final static String PRODUCT_GUID = "0100e712ef4f41bda6258b7d70f62bfa";
+
+    public static final String FALSE = "false";
+
+    /**
+     * 购物车缓存前缀
+     */
+    public static final String CUSTOMER_CAR_PREFIX="customer_car:";
+
+    /**
+     * 微信API调用token
+     */
+    public static final String WECHAT_API_TOKEN="weChatApiAccessToken";
+    /**
+     * 微信授权信心mapkey
+     */
+    public static final String WECHAT_INFO_PREFIX="weChatInfoPrefix_";
+
+    public static final String WECHAT_INFO="wechatUserInfo";
+}

+ 54 - 0
UtilsDemo-WX/src/main/java/com/example/controller/OrderController.java

@@ -0,0 +1,54 @@
+package com.example.controller;
+
+import com.alibaba.fastjson.JSON;
+import com.example.base.*;
+import com.example.entity.OmsOrder;
+import com.example.enums.PayStatusEnum;
+import com.example.enums.ResultEnum;
+import com.example.util.GeneratorIdUtils;
+import com.example.util.RedisUtil;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+import java.math.BigDecimal;
+import java.util.Date;
+
+@Slf4j
+@RestController
+@RequestMapping("/order")
+@Api(tags = "创建订单")
+public class OrderController extends BaseController {
+
+    private final static String OMSORDER = "oms_order_";
+
+    @Autowired
+    private RedisUtil redisUtil;
+
+    @ApiOperation(value = "创建订单")
+    @PostMapping("/createOrder")
+    public ResponseBase createOrder() {
+        GeneratorIdUtils generatorIdUtils = new GeneratorIdUtils();
+        String orderId = generatorIdUtils.nextId();
+        OmsOrder order = new OmsOrder();
+        order.setId(orderId);
+        order.setProductId("23");
+        order.setCreateDate(new Date());
+        order.setStatus(PayStatusEnum.INIT.getIndex());
+        order.setOrderId(orderId);
+        order.setPayAmount(new BigDecimal(1));
+        order.setDiscountAmount(new BigDecimal(0));
+        order.setTotalAmount(new BigDecimal(1));
+        order.setProductName("商品测试");
+        order.setUserId(111L);
+        redisUtil.set(OMSORDER + orderId, order);
+        OmsOrder omsOrder = JSON.parseObject(redisUtil.get(OMSORDER + orderId).toString(),OmsOrder.class);
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS,omsOrder));
+    }
+
+
+}

+ 146 - 0
UtilsDemo-WX/src/main/java/com/example/controller/WechatAppController.java

@@ -0,0 +1,146 @@
+package com.example.controller;
+
+import cn.binarywang.wx.miniapp.api.WxMaService;
+import cn.binarywang.wx.miniapp.bean.WxMaJscode2SessionResult;
+import cn.binarywang.wx.miniapp.bean.WxMaPhoneNumberInfo;
+import cn.binarywang.wx.miniapp.bean.WxMaUserInfo;
+import com.alibaba.fastjson.JSON;
+import com.alibaba.fastjson.JSONArray;
+import com.alibaba.fastjson.JSONObject;
+import com.example.base.BaseController;
+import com.example.base.ResponseBase;
+import com.example.base.ResultVO;
+import com.example.entity.LoginRequest;
+import com.example.entity.MyWxMpUser;
+import com.example.enums.ResultEnum;
+import com.example.util.WxUtils;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.common.error.WxErrorException;
+import org.apache.commons.lang3.StringUtils;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Controller;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestBody;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.ResponseBody;
+
+import java.util.Objects;
+
+@Slf4j
+@Controller
+@RequestMapping("/wechat/app")
+@Api(tags = "微信小程序api服务")
+public class WechatAppController  extends BaseController {
+    private final Logger logger = LoggerFactory.getLogger(this.getClass());
+
+
+    @Autowired
+    private WxMaService wxMaService;
+
+
+
+    /**
+     * 登陆接口
+     */
+    @ApiOperation(value = "微信登录", notes = "微信登录")
+    @PostMapping("/login")
+    @ResponseBody
+    public ResponseBase login(@RequestBody LoginRequest request) {
+        String code = request.getCode();
+        if (StringUtils.isBlank(code)) {
+            return responseError(500,"请输入code错误!");
+        }
+        try {
+            // 获取微信用户session
+            WxMaJscode2SessionResult session = wxMaService.getUserService().getSessionInfo(request.getCode());
+            if (null == session) {
+                throw new RuntimeException("login handler error");
+            }
+            WxMaUserInfo wxUserInfo = JSON.parseObject(request.getRawData(), WxMaUserInfo.class);
+            if (null == wxUserInfo) {
+                throw new RuntimeException("wxUser not exist");
+            }
+
+            MyWxMpUser myWxMpUser = new MyWxMpUser();
+            //TODO 可以增加自己的逻辑,关联业务相关数据
+            myWxMpUser.setOpenId(session.getOpenid());
+            myWxMpUser.setUnionId(session.getUnionid());
+            myWxMpUser.setNickname(wxUserInfo.getNickName());
+            myWxMpUser.setHeadImgUrl(wxUserInfo.getAvatarUrl());
+            myWxMpUser.setSex(Integer.parseInt(wxUserInfo.getGender()));
+            //后台新增用户成功
+//            Map<String, String> tokenMap = customerInfoService.wxlogin(myWxMpUser);
+            //登录系统获取本地
+//            return R.success(tokenMap);
+            return responseSuccess(new ResultVO());
+        } catch (WxErrorException e) {
+            this.logger.error(e.getMessage(), e);
+            return responseError(500,"登录错误!");
+        }
+    }
+
+//    @Autowired
+//    private CustomerService customerService;
+
+    /**
+     * 登陆接口
+     */
+    @ApiOperation(value = "小程序绑定手机号", notes = "小程序绑定手机号")
+    @PostMapping("/login/phone")
+    @ResponseBody
+    public ResponseBase loginphone(@RequestBody LoginRequest request) throws WxErrorException {
+        // 获取微信用户session
+        WxMaJscode2SessionResult session = wxMaService.getUserService().getSessionInfo(request.getCode());
+        //解密手机号
+        WxMaPhoneNumberInfo phoneNoInfo = wxMaService.getUserService().getPhoneNoInfo(session.getSessionKey(), request.getEncryptedData(), request.getIv());
+
+        //todo:各业务逻辑处理
+//        CustomerInfo customerInfo = new CustomerInfo();
+//        CustomerInfoForm tokenUser = customerService.getCurrentMember();
+//        customerInfo.setGuid(tokenUser.getGuid());
+//        customerInfo.setPhone(phoneNoInfo.getPhoneNumber());
+//        int i = customerInfoService.updateByPrimaryKeySelective(customerInfo);
+//        if (i == 0) {
+//            return resultVOError();
+//        }
+        //登录系统获取本地
+        return responseSuccess(new ResultVO());
+    }
+
+    /**
+     * 登陆接口
+     */
+    @ApiOperation(value = "小程序获取微信步数", notes = "小程序获取微信步数")
+    @PostMapping("/steps")
+    @ResponseBody
+    public ResponseBase steps(@RequestBody LoginRequest request) throws WxErrorException {
+        // 获取微信用户session
+        WxMaJscode2SessionResult session = wxMaService.getUserService().getSessionInfo(request.getCode());
+        String decrypt = null;
+        //解密
+        try {
+            decrypt = WxUtils.decryptWeChatRunInfo(session.getSessionKey(), request.getEncryptedData(), request.getIv());
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+
+        if (decrypt == null) {
+            log.info("解密失败");
+        } else {
+            log.info("解析成功");
+        }
+        JSONObject stepInfoListJson = JSON.parseObject(decrypt);
+        //解析json,获取stepInfoList下面的步数
+        JSONArray stepInfoList = JSON.parseArray(stepInfoListJson.getString("stepInfoList"));
+        //获取今天的步数
+        JSONObject today = Objects.requireNonNull(stepInfoList).getJSONObject(0);
+        //登录系统获取本地
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS,today));
+    }
+
+
+}

+ 233 - 0
UtilsDemo-WX/src/main/java/com/example/controller/WechatAppPayController.java

@@ -0,0 +1,233 @@
+package com.example.controller;
+
+import com.alibaba.fastjson.JSON;
+import com.example.base.*;
+import com.example.entity.OmsOrder;
+import com.example.entity.WxPayFrom;
+import com.example.enums.PayStatusEnum;
+import com.example.enums.ResultEnum;
+import com.example.util.GeneratorIdUtils;
+import com.example.util.RedisUtil;
+import com.example.util.UUIDUtils;
+import com.github.binarywang.wxpay.bean.notify.WxPayOrderNotifyResult;
+import com.github.binarywang.wxpay.bean.request.WxPayUnifiedOrderRequest;
+import com.github.binarywang.wxpay.constant.WxPayConstants;
+import com.github.binarywang.wxpay.exception.WxPayException;
+import com.github.binarywang.wxpay.service.WxPayService;
+import io.swagger.annotations.*;
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.common.bean.WxJsapiSignature;
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.api.WxMpService;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.stereotype.Controller;
+import org.springframework.web.bind.annotation.*;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.PrintWriter;
+import java.math.BigDecimal;
+
+import static com.alibaba.fastjson.JSON.parseObject;
+
+@Slf4j
+@Controller
+@RequestMapping("/wechat/pay")
+@Api(tags = "微信小程序支付服务")
+public class WechatAppPayController extends BaseController {
+    private final static String OMSORDER = "oms_order_";
+
+    @Autowired
+    private WxMpService wxMpService;
+
+    @Autowired
+    private WxPayService wxPayService;
+
+
+
+    @Value("${wx.pay.wxpayAppcallbackurl}")
+    private String wxpayCallbackUrl;
+
+    @Autowired
+    private RedisUtil redisUtil;
+
+    @ApiOperation(value = "微信支付接口(二次支付)", notes = "微信支付接口请勿随意调用(二次支付)")
+    @ApiResponse(code = 500, message = "请求的配置参数有问题")
+//    @AuthCarbonValidate
+    @PostMapping
+    @ResponseBody
+    public ResponseBase pay(@RequestBody WxPayFrom wxPayFrom, HttpServletRequest request) {
+        String orderId = wxPayFrom.getOrderId();
+        String openid = wxPayFrom.getOpenid();
+        String tradeType = wxPayFrom.getTradeType();
+        OmsOrder omsOrder = parseObject(redisUtil.get(OMSORDER + orderId).toString(),OmsOrder.class);
+        Integer status = omsOrder.getStatus();
+        //查询订单是否支付
+        if (status.equals(PayStatusEnum.SUCCESS.getIndex())) {
+            throw new RuntimeException("订单已支付");
+        }
+        WxPayUnifiedOrderRequest orderRequest = new WxPayUnifiedOrderRequest();
+        //随机字符串
+        String s = UUIDUtils.randomUUID();
+
+        orderRequest.setNonceStr(s);
+        //商品描述
+        orderRequest.setBody("购碳扶贫");
+        //商户订单号(支付编号)
+        GeneratorIdUtils generatorIdUtils = new GeneratorIdUtils();
+        String orderNum = generatorIdUtils.nextId();
+        log.info("orderNum:" + orderNum);
+        //设置商户订单号
+        orderRequest.setOutTradeNo(orderNum);
+        //设置金额
+        //data.getTotalPrice();
+        BigDecimal money = omsOrder.getPayAmount();
+        //元转成分
+        money = money.multiply(new BigDecimal(100));
+        int i = money.intValue();
+        if ("test".equals(tradeType)) {
+            //测试环境一分钱
+            orderRequest.setTotalFee(1);
+        } else {
+            orderRequest.setTotalFee(i);
+        }
+        //终端ip
+        String ipAddress = getIpAddress(request);
+        orderRequest.setSpbillCreateIp(ipAddress);
+        // 通知地址 支付成功后回调地址
+        log.info("wxpayCallbackUrl=================>{}", wxpayCallbackUrl);
+        orderRequest.setNotifyUrl(wxpayCallbackUrl);
+        //交易类型
+        orderRequest.setTradeType(WxPayConstants.TradeType.JSAPI);
+        //用户标识 获取用户openid
+        orderRequest.setOpenid(openid);
+        // 这个可能是偏向原生一点的统一下单,返回的参数有很多没用的 或者null值 建议使用 createOrder下单
+        Object unOrder = null;
+        try {
+            unOrder = wxPayService.createOrder(orderRequest);
+        } catch (WxPayException e) {
+            //e.printStackTrace();
+            log.error(e.getMessage());
+            //签名完成新增支付订单
+            return responseError(500,"微信支付失败");
+        }
+        //更新本地订单支付id
+        //orderClient.update(data);
+//        orderInfoService.update(goodsOrderInfo, orderNum);
+        //签名完成新增支付订单
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS,unOrder));
+    }
+
+
+    @ApiOperation(value = "微信支付结果回调", notes = "微信支付结果回调请勿直接调用")
+//    @AuthCarbonValidate
+    @PostMapping("/payResultNotified")
+    public void payOrderResultNotify(HttpServletRequest request, HttpServletResponse response) throws IOException {
+        log.info("微信支付返回通知函数开始---------------------");
+
+        InputStream inStream = request.getInputStream();
+        ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
+        byte[] buffer = new byte[1024];
+        int len = 0;
+        while ((len = inStream.read(buffer)) != -1) {
+            outSteam.write(buffer, 0, len);
+        }
+        outSteam.close();
+        inStream.close();
+        String result = new String(outSteam.toByteArray(), "utf-8");
+        boolean isPayOk = false;
+        WxPayOrderNotifyResult wxPayOrderNotifyResult = null;
+        // 此处调用订单查询接口验证是否交易成功
+        try {
+            wxPayOrderNotifyResult = wxPayService.parseOrderNotifyResult(result);
+            if ("SUCCESS".equals(wxPayOrderNotifyResult.getResultCode())) {
+                isPayOk = true;
+            }
+            log.info("解析数据:" + wxPayOrderNotifyResult.toString());
+        } catch (WxPayException e) {
+            e.printStackTrace();
+        }
+        String noticeStr = "";
+        // 支付成功,商户处理后同步返回给微信参数
+        PrintWriter writer = response.getWriter();
+        if (isPayOk) {
+            //建议在这里处理付款完成的业务(虽然前端也可以处理后续业务,但是前端处理并不安全,例如:客户突然关闭浏览器了等情况,付款成功后续的业务将中断)
+            //支付订单支付编号
+            System.out.println("===============付款成功,业务开始处理==============");
+            String orderNum = wxPayOrderNotifyResult.getOutTradeNo();
+            log.info("orderNum:" + orderNum);
+            OmsOrder omsOrder = JSON.parseObject(redisUtil.get(OMSORDER + orderNum).toString(),OmsOrder.class);
+            omsOrder.setStatus(PayStatusEnum.SUCCESS.getIndex());
+            redisUtil.set(OMSORDER + omsOrder.getOrderId(), omsOrder);
+            //将订单更新为支付
+            // 更新订单信息(含商品订单,支付订单信息)
+//            int i = orderInfoService.updateFullOrderInfo(orderNum, 2);
+            /**
+             * 自己的业务
+             */
+            System.out.println("===============付款成功,业务处理完毕==============");
+            // 通知微信已经收到消息,不要再给我发消息了,否则微信会8连击调用本接口
+            noticeStr = setXML("SUCCESS", "OK");
+            log.info("收到通知返回给微信api信息:-----------" + noticeStr);
+            writer.write(noticeStr);
+            writer.flush();
+        } else {
+            // 支付失败, 记录流水失败
+            noticeStr = setXML("FAIL", "");
+            writer.write(noticeStr);
+            writer.flush();
+            System.out.println("===============支付失败==============");
+        }
+    }
+
+    public static String setXML(String return_code, String return_msg) {
+        return "<xml><return_code><![CDATA[" + return_code + "]]></return_code><return_msg><![CDATA[" + return_msg + "]]></return_msg></xml>";
+    }
+
+    public static String getIpAddress(HttpServletRequest request) {
+        String ip = request.getHeader("x-forwarded-for");
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("WL-Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_CLIENT_IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_X_FORWARDED_FOR");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getRemoteAddr();
+        }
+        String[] split = ip.split(",");
+        ip = split[0];
+        return ip;
+    }
+
+
+    @ApiOperation(value = "微信js签名接口", notes = "微信js签名接口")
+    @ApiResponses({
+            @ApiResponse(code = 0, message = "成功")
+    })
+//    @AuthCarbonValidate
+    @GetMapping("getJsapiTicket")
+    public ResponseBase getJsapiTicket(String url) {
+        WxJsapiSignature jsapiSignature = null;
+        try {
+            jsapiSignature = wxMpService.createJsapiSignature(url);
+
+        } catch (WxErrorException e) {
+            e.printStackTrace();
+            System.err.print(e.getError());
+        }
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS,jsapiSignature));
+    }
+}
+
+

+ 205 - 0
UtilsDemo-WX/src/main/java/com/example/controller/WechatH5Controller.java

@@ -0,0 +1,205 @@
+package com.example.controller;
+
+import com.alibaba.fastjson.JSONObject;
+import com.example.base.BaseController;
+import com.example.base.ResponseBase;
+import com.example.base.ResultVO;
+import com.example.base.Token;
+import com.example.constant.Constants;
+import com.example.constant.GlobleConstant;
+import com.example.entity.User;
+import com.example.enums.ResultEnum;
+import com.example.service.UserService;
+import com.example.util.RedisUtil;
+import com.example.util.TokenUtil;
+import com.example.vo.UserUpdateVO;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.common.api.WxConsts;
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.api.WxMpService;
+import me.chanjar.weixin.mp.bean.result.WxMpOAuth2AccessToken;
+import me.chanjar.weixin.mp.bean.result.WxMpUser;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.context.ApplicationContext;
+import org.springframework.stereotype.Controller;
+import org.springframework.util.StringUtils;
+import org.springframework.web.bind.annotation.*;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.IOException;
+import java.net.URLEncoder;
+
+@Slf4j
+@Controller
+@RequestMapping("/wechat/h5")
+@Api(tags = "公众号网页登录")
+public class WechatH5Controller extends BaseController {
+
+    private static final String WX_USER = "wx_user_";
+
+    /**
+     * h5
+     */
+    @Autowired
+    private WxMpService wxMpService;
+
+    @Value("${jwt.tokenHeader}")
+    private String tokenHeader;
+    @Value("${jwt.tokenHead}")
+    private String tokenHead;
+
+
+    @Autowired
+    private RedisUtil redisUtil;
+
+    @Autowired
+    private UserService userService;
+
+
+    private static final String WECHAT_USER_INFO = "wechatUserInfo";
+
+
+    @Value("${wx.mp.url}")//本地后台地址
+    private String url;
+    @Value("${wx.mp.before}")//前端首页地址
+    private String reUrl;
+
+    /**
+     * 请求微信服务器认证授权
+     *
+     * @param returnUrl 微信认证授权时的state参数
+     * @return
+     */
+
+
+    @ApiOperation(value = "微信认证")
+    @GetMapping("/authorize")
+    public void authorize(@RequestParam("returnUrl") String returnUrl, HttpServletResponse response) throws IOException {
+        if (!StringUtils.isEmpty(returnUrl)) {
+            reUrl = returnUrl;
+        }
+        // 用户授权完成后的重定向链接,441aca54.ngrok.io为使用ngrok代理后的公网域名,该域名需要替换为在微信公众号后台设置的域名,否则请求微信服务器不成功,一般都是采用将本地服务代理映射到一个可以访问的公网进行开发调试
+        String redirectURL = wxMpService.oauth2buildAuthorizationUrl(url, WxConsts.OAuth2Scope.SNSAPI_USERINFO, URLEncoder.encode(returnUrl, "utf-8"));
+        log.info("【微信网页授权】获取code,redirectURL={}", redirectURL);
+//        redirectURL ="https://www.baidu.com/";
+        response.sendRedirect(redirectURL);
+    }
+
+    @Autowired
+    ApplicationContext applicationContext;
+
+    /**
+     * 认证授权成功后返回微信用户信息
+     *
+     * @param code      微信授权成功后重定向地址后的code参数
+     * @param returnUrl 请求微信授权时携带的state参数
+     * @return
+     * @throws Exception
+     */
+    @ApiOperation(value = "微信认证成功后回调")
+    @GetMapping("/userInfo")
+    public String userInfo(@RequestParam("code") String code,
+                           @RequestParam("state") String returnUrl) throws Exception {
+        log.info("【微信网页授权】code={}", code);
+        log.info("【微信网页授权】state={}", returnUrl);
+        WxMpOAuth2AccessToken wxMpOAuth2AccessToken;
+        try {
+            // 获取accessToken
+            wxMpOAuth2AccessToken = wxMpService.oauth2getAccessToken(code);
+        } catch (WxErrorException e) {
+            log.error("【微信网页授权】{}", e);
+            throw new Exception(e.getError().getErrorMsg());
+        }
+        // 根据accessToken获取openId
+        String openId = wxMpOAuth2AccessToken.getOpenId();
+        log.info("【微信网页授权】openId={}", openId);
+        //applicationContext.publishEvent(new LoginOutEvent(this,wxMpOAuth2AccessToken.getAccessToken()));
+        // 获取用户信息
+        WxMpUser wxMpUser = wxMpService.oauth2getUserInfo(wxMpOAuth2AccessToken, null);
+        log.info("【微信网页授权】wxMpUser={}", wxMpUser);
+        // 刷新accessToken
+        wxMpService.oauth2refreshAccessToken(wxMpOAuth2AccessToken.getRefreshToken());
+        // 验证accessToken是否有效
+        wxMpService.oauth2validateAccessToken(wxMpOAuth2AccessToken);
+
+        redisUtil.set(WX_USER + openId, wxMpUser);
+
+        User user = userService.getUserByOpenId(wxMpUser.getOpenId());
+        if (user == null) {
+            userService.addUser(wxMpUser);
+        }
+
+        Token token = TokenUtil.createJwtToken(Constants.LOGIN_TOKEN_KEY+user.getOpenId(), GlobleConstant.CURRENT_MEMBER);
+        String accessToken = token.getAccessToken();
+        System.err.println("生成的token:"+accessToken);
+        redisUtil.set(accessToken, wxMpUser);
+//        resultMap.put("accessToken", accessToken);
+
+//        tokenService.createToken();
+
+
+
+
+//        UmsMemberPlatform umsMemberPlatform= new UmsMemberPlatform();
+//        String nickname = wxMpUser.getNickname();
+//        wxMpUser.setNickname(EmojiUtil.filterEmoji(nickname));
+//        BeanUtil.copyProperties(wxMpUser,umsMemberPlatform);
+//        umsMemberPlatform.setOpenid(wxMpUser.getOpenId());
+//        umsMemberPlatform.setUnionid(wxMpUser.getUnionId());
+//        umsMemberPlatform.setHeadimgurl(wxMpUser.getHeadImgUrl());
+//
+//        String token = umsMemberPlatformService.addUmsMemberPlatformService(umsMemberPlatform);
+        String trueUrl = null;
+//        if (StringUtils.isEmpty(returnUrl)){
+//            trueUrl= reUrl+"?token="+token+"&tokenHead="+tokenHead;
+//        }else if (!returnUrl.contains("?")){
+//            trueUrl=returnUrl+"?token="+token+"&tokenHead="+tokenHead;
+//        }else {
+//            trueUrl=returnUrl+"&token="+token+"&tokenHead="+tokenHead;
+//        }
+        return "redirect:" + returnUrl + "?token=" + accessToken;
+
+    }
+
+    @GetMapping("/user/get")
+    @ApiOperation("获取微信用户信息")
+    @ResponseBody
+    public ResponseBase getUserInfo(HttpServletRequest request) {
+        String openId = TokenUtil.parseJWT(request.getHeader("Authorization")).get("uid").toString().replace(Constants.LOGIN_TOKEN_KEY,"");
+        User user = userService.getUserByOpenId(openId);
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS, user));
+    }
+
+    @GetMapping("/code/{tel}")
+    @ApiOperation("获取验证码")
+    @ResponseBody
+    public ResponseBase getCode(@PathVariable(value = "tel") String tel) {
+        JSONObject json = new JSONObject();
+        String code = (int) (Math.random() * 9000 + 1000)+"";
+        json.put("code", code);
+        redisUtil.set("phone_code_"+tel,code,300000);
+        return responseSuccess(success(json));
+    }
+
+    @PostMapping("/user/update")
+    @ApiOperation("修改用户手机号")
+    @ResponseBody
+    public ResponseBase updateUserPhone(@RequestBody UserUpdateVO updateVO,HttpServletRequest request){
+        String openId = TokenUtil.parseJWT(request.getHeader("Authorization")).get("uid").toString().replace(Constants.LOGIN_TOKEN_KEY,"");
+        String checkCode = (String) redisUtil.get("phone_code_"+updateVO.getPhone());
+        if(StringUtils.isEmpty(checkCode)){
+            return responseError(1,"验证码错误或已过期");
+        }
+        if(!checkCode.equals(updateVO.getCode())){
+            return responseError(1,"验证码不匹配");
+        }
+        redisUtil.del("phone_code_"+updateVO.getPhone());
+        userService.updateUser(openId,updateVO);
+        return responseSuccess(success());
+    }
+
+}

+ 337 - 0
UtilsDemo-WX/src/main/java/com/example/controller/WechatH5PayController.java

@@ -0,0 +1,337 @@
+package com.example.controller;
+
+import com.alibaba.fastjson.JSON;
+import com.example.base.*;
+import com.example.entity.OmsOrder;
+import com.example.entity.RefundOrder;
+import com.example.entity.WxPayFrom;
+import com.example.enums.PayStatusEnum;
+import com.example.enums.ResultEnum;
+import com.example.util.GeneratorIdUtils;
+import com.example.util.RedisUtil;
+import com.example.util.UUIDUtils;
+import com.github.binarywang.wxpay.bean.notify.WxPayOrderNotifyResult;
+import com.github.binarywang.wxpay.bean.notify.WxPayRefundNotifyResult;
+import com.github.binarywang.wxpay.bean.request.WxPayRefundRequest;
+import com.github.binarywang.wxpay.bean.request.WxPayUnifiedOrderRequest;
+import com.github.binarywang.wxpay.bean.result.WxPayOrderQueryResult;
+import com.github.binarywang.wxpay.bean.result.WxPayRefundQueryResult;
+import com.github.binarywang.wxpay.constant.WxPayConstants;
+import com.github.binarywang.wxpay.exception.WxPayException;
+import com.github.binarywang.wxpay.service.WxPayService;
+import io.swagger.annotations.Api;
+import io.swagger.annotations.ApiOperation;
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.common.bean.WxJsapiSignature;
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.api.WxMpService;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.beans.factory.annotation.Value;
+import org.springframework.web.bind.annotation.*;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
+import java.io.InputStream;
+import java.io.PrintWriter;
+import java.math.BigDecimal;
+
+@Slf4j
+@RestController
+@Api(tags = "公众号支付", description = "WechatH5PayController")
+@RequestMapping("/app")
+public class WechatH5PayController extends BaseController {
+
+    private final static String OMSORDER = "oms_order_";
+
+    @Value("${wx.pay.wxpayAppcallbackurl}")
+    private String wxpayCallbackUrl;
+
+    @Value("${wx.pay.wxpayRefundCallBackUrl}")
+    private String wxpayRefundCallBackUrl;
+
+    @Autowired
+    private WxPayService wxPayService;
+
+    @Autowired
+    private RedisUtil redisUtil;
+
+
+    @PostMapping("/pay")
+    public ResponseBase pay(@RequestBody WxPayFrom wxPayFrom, HttpServletRequest request) {
+        String orderId = wxPayFrom.getOrderId();
+        String openid = wxPayFrom.getOpenid();
+        String tradeType = wxPayFrom.getTradeType();
+        OmsOrder omsOrder = JSON.parseObject(redisUtil.get(OMSORDER + orderId).toString(), OmsOrder.class);
+        Integer status = omsOrder.getStatus();
+        //查询订单是否支付
+        if (status.equals(PayStatusEnum.SUCCESS.getIndex())) {
+            throw new RuntimeException("订单已支付");
+        }
+        WxPayUnifiedOrderRequest orderRequest = new WxPayUnifiedOrderRequest();
+        //随机字符串
+        String s = UUIDUtils.randomUUID();
+
+        orderRequest.setNonceStr(s);
+        //商品描述
+        String productName = omsOrder.getProductName();
+        orderRequest.setBody(productName);
+        //商户订单号(支付编号)
+//        GeneratorIdUtils generatorIdUtils = new GeneratorIdUtils();
+//        String orderNum = generatorIdUtils.nextId();
+//        log.info("orderNum:" + orderNum);
+        //设置商户订单号
+        orderRequest.setOutTradeNo(orderId);
+        //设置金额
+        //data.getTotalPrice();
+        BigDecimal money = omsOrder.getPayAmount();
+        //元转成分
+        money = money.multiply(new BigDecimal(100));
+        int i = money.intValue();
+        if ("test".equals(tradeType)) {
+            //测试环境一分钱
+            orderRequest.setTotalFee(1);
+        } else {
+            orderRequest.setTotalFee(i);
+        }
+        //终端ip
+        String ipAddress = getIpAddress(request);
+        orderRequest.setSpbillCreateIp(ipAddress);
+        // 通知地址 支付成功后回调地址
+        log.info("wxpayCallbackUrl=================>{}", wxpayCallbackUrl);
+        orderRequest.setNotifyUrl(wxpayCallbackUrl);
+        //交易类型
+        orderRequest.setTradeType(WxPayConstants.TradeType.JSAPI);
+        //用户标识 获取用户openid
+        orderRequest.setOpenid(openid);
+        // 这个可能是偏向原生一点的统一下单,返回的参数有很多没用的 或者null值 建议使用 createOrder下单
+        Object unOrder = null;
+        try {
+            unOrder = wxPayService.createOrder(orderRequest);
+        } catch (WxPayException e) {
+            //e.printStackTrace();
+            log.error(e.getMessage());
+            //签名完成新增支付订单
+            return responseError(500, "微信支付失败");
+        }
+        //更新本地订单支付id
+        //orderClient.update(data);
+
+        //签名完成新增支付订单
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS, unOrder));
+    }
+
+    @Autowired
+    private WxMpService wxMpService;
+
+    /*微信js签名接口*/
+    @GetMapping("/getJsapiTicket")
+    public ResponseBase getJsapiTicket(String url) {
+        WxJsapiSignature jsapiSignature = null;
+        try {
+            jsapiSignature = wxMpService.createJsapiSignature(url);
+        } catch (WxErrorException e) {
+            e.printStackTrace();
+            System.err.print(e.getError());
+        }
+        return responseSuccess(new ResultVO(ResultEnum.SUCCESS, jsapiSignature));
+
+    }
+
+    /**
+     * wx支付成功回调接口
+     *
+     * @param request
+     * @param response
+     * @throws IOException
+     */
+    @RequestMapping("/payResult")
+    @ApiOperation("微信支付返回通知")
+    public void payResult(HttpServletRequest request, HttpServletResponse response) throws IOException {
+        log.info("微信支付返回通知函数开始---------------------");
+
+        InputStream inStream = request.getInputStream();
+        ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
+        byte[] buffer = new byte[1024];
+        int len = 0;
+        while ((len = inStream.read(buffer)) != -1) {
+            outSteam.write(buffer, 0, len);
+        }
+        outSteam.close();
+        inStream.close();
+        String result = new String(outSteam.toByteArray(), "utf-8");
+        boolean isPayOk = false;
+        WxPayOrderNotifyResult wxPayOrderNotifyResult = null;
+        // 此处调用订单查询接口验证是否交易成功
+        try {
+            wxPayOrderNotifyResult = wxPayService.parseOrderNotifyResult(result);
+            if ("SUCCESS".equals(wxPayOrderNotifyResult.getResultCode())) {
+                isPayOk = true;
+            }
+            log.info("解析数据:" + wxPayOrderNotifyResult.toString());
+        } catch (WxPayException e) {
+            e.printStackTrace();
+        }
+        String noticeStr = "";
+        // 支付成功,商户处理后同步返回给微信参数
+        PrintWriter writer = response.getWriter();
+        if (isPayOk) {
+            //建议在这里处理付款完成的业务(虽然前端也可以处理后续业务,但是前端处理并不安全,例如:客户突然关闭浏览器了等情况,付款成功后续的业务将中断)
+            //支付订单支付编号
+            System.out.println("===============付款成功,业务开始处理==============");
+            String orderNum = wxPayOrderNotifyResult.getOutTradeNo();
+            log.info("orderNum:" + orderNum);
+            //将订单更新为支付
+            OmsOrder omsOrder = JSON.parseObject(redisUtil.get(OMSORDER + orderNum).toString(), OmsOrder.class);
+            omsOrder.setStatus(PayStatusEnum.SUCCESS.getIndex());
+            redisUtil.set(OMSORDER + omsOrder.getOrderId(), omsOrder);
+            /**
+             * 自己的业务
+             */
+            System.out.println("===============付款成功,业务处理完毕==============");
+            // 通知微信已经收到消息,不要再给我发消息了,否则微信会8连击调用本接口
+            noticeStr = setXML("SUCCESS", "OK");
+            log.info("收到通知返回给微信api信息:-----------" + noticeStr);
+            writer.write(noticeStr);
+            writer.flush();
+        } else {
+            // 支付失败, 记录流水失败
+            noticeStr = setXML("FAIL", "");
+            writer.write(noticeStr);
+            writer.flush();
+            System.out.println("===============支付失败==============");
+        }
+    }
+
+
+    public static String setXML(String return_code, String return_msg) {
+        return "<xml><return_code><![CDATA[" + return_code + "]]></return_code><return_msg><![CDATA[" + return_msg + "]]></return_msg></xml>";
+    }
+
+    public static String getIpAddress(HttpServletRequest request) {
+        String ip = request.getHeader("x-forwarded-for");
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("WL-Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_CLIENT_IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getHeader("HTTP_X_FORWARDED_FOR");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
+            ip = request.getRemoteAddr();
+        }
+        String[] split = ip.split(",");
+        ip = split[0];
+        return ip;
+    }
+
+    @GetMapping("/order/{orderId}")
+    @ApiOperation("查询订单状态")
+    public ResponseBase queryOrder(@PathVariable(value = "orderId") String orderId) {
+        try {
+            WxPayOrderQueryResult orderQueryResult = wxPayService.queryOrder(null, orderId);
+            return responseSuccess(new ResultVO(ResultEnum.SUCCESS, orderQueryResult));
+        } catch (WxPayException e) {
+            e.printStackTrace();
+            return responseError(500, "查询订单异常");
+        }
+    }
+
+    @PostMapping("/order/refund")
+    @ApiOperation("申请退款")
+    public ResponseBase refundOrder(@RequestBody RefundOrder refundOrder) {
+        WxPayRefundRequest refundRequest = new WxPayRefundRequest();
+        try {
+            OmsOrder omsOrder = JSON.parseObject(redisUtil.get(OMSORDER + refundOrder.getOrderId()).toString(), OmsOrder.class);
+            refundRequest.setOutTradeNo(refundOrder.getOrderId());
+            GeneratorIdUtils generatorIdUtils = new GeneratorIdUtils();
+            String refundNo = generatorIdUtils.nextId();
+            refundRequest.setOutRefundNo(refundNo);
+            refundRequest.setRefundDesc(refundOrder.getRefundDesc());
+            refundRequest.setTotalFee(omsOrder.getTotalAmount().intValue());
+            refundRequest.setRefundFee(refundRequest.getTotalFee());
+            refundRequest.setOpUserId(refundRequest.getMchId());
+            refundRequest.setNotifyUrl(wxpayRefundCallBackUrl);
+            wxPayService.refund(refundRequest);
+        } catch (WxPayException e) {
+            e.printStackTrace();
+            return responseError(500, "申请退款异常");
+        }
+        return responseSuccess(new ResultVO());
+    }
+
+
+    @RequestMapping("/refundResult")
+    @ApiOperation("退款回调")
+    public void refundResult(HttpServletRequest request, HttpServletResponse response) throws IOException {
+        log.info("微信退款回调通知函数开始---------------------");
+
+        InputStream inStream = request.getInputStream();
+        ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
+        byte[] buffer = new byte[1024];
+        int len = 0;
+        while ((len = inStream.read(buffer)) != -1) {
+            outSteam.write(buffer, 0, len);
+        }
+        outSteam.close();
+        inStream.close();
+        String result = new String(outSteam.toByteArray(), "utf-8");
+        boolean refundOk = false;
+        WxPayRefundNotifyResult wxPayRefundNotifyResult = null;
+        try {
+            wxPayRefundNotifyResult = wxPayService.parseRefundNotifyResult(result);
+            if ("SUCCESS".equals(wxPayRefundNotifyResult.getReturnCode())) {
+                refundOk = true;
+            }
+            log.info("解析数据:" + wxPayRefundNotifyResult.toString());
+        } catch (WxPayException e) {
+            e.printStackTrace();
+        }
+        String noticeStr = "";
+        // 支付成功,商户处理后同步返回给微信参数
+        PrintWriter writer = response.getWriter();
+        if (refundOk) {
+            //建议在这里处理付款完成的业务(虽然前端也可以处理后续业务,但是前端处理并不安全,例如:客户突然关闭浏览器了等情况,付款成功后续的业务将中断)
+            //支付订单支付编号
+            System.out.println("===============退款成功,业务开始处理==============");
+            String refundNo = wxPayRefundNotifyResult.getReqInfo().getOutRefundNo();
+            log.info("refundNo:" + refundNo);
+            //将订单更新已退款
+            /**
+             * 自己的业务
+             */
+            System.out.println("===============退款成功,业务处理完毕==============");
+            //todo:关闭订单
+            noticeStr = setXML("SUCCESS", "OK");
+            log.info("收到通知返回给微信api信息:-----------" + noticeStr);
+            writer.write(noticeStr);
+            writer.flush();
+        } else {
+            // 支付失败, 记录流水失败
+            noticeStr = setXML("FAIL", "");
+            writer.write(noticeStr);
+            writer.flush();
+            System.out.println("===============退款失败==============");
+        }
+    }
+
+    @ApiOperation("退款订单查询")
+    @GetMapping("/order/refund/{orderId}")
+    public ResponseBase refundQuery(@PathVariable(value = "orderId") String orderId) {
+        try {
+            WxPayRefundQueryResult refundQueryResult = wxPayService.refundQuery(null, orderId, null, null);
+        } catch (WxPayException e) {
+            e.printStackTrace();
+            return responseError(500, "查询退款异常");
+        }
+        return responseSuccess(new ResultVO());
+    }
+
+
+}

+ 29 - 0
UtilsDemo-WX/src/main/java/com/example/entity/LoginRequest.java

@@ -0,0 +1,29 @@
+package com.example.entity;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+@Data
+@ApiModel("小程序提交参数")
+public class LoginRequest {
+    //用户登录凭证
+    @ApiModelProperty("用户登录凭证")
+    String code;
+
+    //原始数据字符串
+    @ApiModelProperty("原始数据字符串")
+    String signature;
+
+    //加密用户数据
+    @ApiModelProperty("加密用户数据")
+    String encryptedData;
+
+    //加密算法的初始向量
+    @ApiModelProperty("加密算法的初始向量")
+    String iv;
+
+    //校验用户信息字符串
+    String rawData;
+
+}

+ 184 - 0
UtilsDemo-WX/src/main/java/com/example/entity/LoginUser.java

@@ -0,0 +1,184 @@
+package com.example.entity;
+
+import com.fasterxml.jackson.annotation.JsonIgnore;
+
+import java.util.Set;
+
+/**
+ * 登录用户身份权限
+ * 
+ * @author ruoyi
+ */
+public class LoginUser
+{
+    private static final long serialVersionUID = 1L;
+
+    /**
+     * 用户唯一标识
+     */
+    private String token;
+
+    /**
+     * 登录时间
+     */
+    private Long loginTime;
+
+    /**
+     * 过期时间
+     */
+    private Long expireTime;
+
+    /**
+     * 登录IP地址
+     */
+    private String ipaddr;
+
+    /**
+     * 登录地点
+     */
+    private String loginLocation;
+
+    /**
+     * 浏览器类型
+     */
+    private String browser;
+
+    /**
+     * 操作系统
+     */
+    private String os;
+
+    /**
+     * 权限列表
+     */
+    private Set<String> permissions;
+
+
+    public String getToken()
+    {
+        return token;
+    }
+
+    public void setToken(String token)
+    {
+        this.token = token;
+    }
+
+    public LoginUser()
+    {
+    }
+
+
+    /**
+     * 账户是否未过期,过期无法验证
+     */
+    @JsonIgnore
+    public boolean isAccountNonExpired()
+    {
+        return true;
+    }
+
+    /**
+     * 指定用户是否解锁,锁定的用户无法进行身份验证
+     * 
+     * @return
+     */
+    @JsonIgnore
+    public boolean isAccountNonLocked()
+    {
+        return true;
+    }
+
+    /**
+     * 指示是否已过期的用户的凭据(密码),过期的凭据防止认证
+     * 
+     * @return
+     */
+    @JsonIgnore
+    public boolean isCredentialsNonExpired()
+    {
+        return true;
+    }
+
+    /**
+     * 是否可用 ,禁用的用户不能身份验证
+     * 
+     * @return
+     */
+    @JsonIgnore
+    public boolean isEnabled()
+    {
+        return true;
+    }
+
+    public Long getLoginTime()
+    {
+        return loginTime;
+    }
+
+    public void setLoginTime(Long loginTime)
+    {
+        this.loginTime = loginTime;
+    }
+
+    public String getIpaddr()
+    {
+        return ipaddr;
+    }
+
+    public void setIpaddr(String ipaddr)
+    {
+        this.ipaddr = ipaddr;
+    }
+
+    public String getLoginLocation()
+    {
+        return loginLocation;
+    }
+
+    public void setLoginLocation(String loginLocation)
+    {
+        this.loginLocation = loginLocation;
+    }
+
+    public String getBrowser()
+    {
+        return browser;
+    }
+
+    public void setBrowser(String browser)
+    {
+        this.browser = browser;
+    }
+
+    public String getOs()
+    {
+        return os;
+    }
+
+    public void setOs(String os)
+    {
+        this.os = os;
+    }
+
+    public Long getExpireTime()
+    {
+        return expireTime;
+    }
+
+    public void setExpireTime(Long expireTime)
+    {
+        this.expireTime = expireTime;
+    }
+
+    public Set<String> getPermissions()
+    {
+        return permissions;
+    }
+
+    public void setPermissions(Set<String> permissions)
+    {
+        this.permissions = permissions;
+    }
+
+}

+ 12 - 0
UtilsDemo-WX/src/main/java/com/example/entity/MyWxMpUser.java

@@ -0,0 +1,12 @@
+package com.example.entity;
+
+import lombok.Data;
+import me.chanjar.weixin.mp.bean.result.WxMpUser;
+
+@Data
+public class MyWxMpUser extends WxMpUser {
+
+    private String code;
+
+    private String phone;
+}

+ 46 - 0
UtilsDemo-WX/src/main/java/com/example/entity/OmsOrder.java

@@ -0,0 +1,46 @@
+package com.example.entity;
+
+import lombok.Data;
+
+import java.io.Serializable;
+import java.math.BigDecimal;
+import java.util.Date;
+
+
+@Data
+public class OmsOrder implements Serializable {
+
+    //主键
+    private String id;
+
+    //订单id
+    private String orderId;
+
+    //商品描述
+    private String productName;
+
+    //商品ID
+    private String productId;
+
+    //支付金额
+    private BigDecimal payAmount;
+
+    //总金额
+    private BigDecimal totalAmount;
+
+    //优惠金额
+    private BigDecimal discountAmount;
+
+    //支付状态
+    private Integer status;
+
+    //订单创建时间
+    private Date createDate;
+
+    //创建人ID
+    private Long userId;
+
+
+
+
+}

+ 20 - 0
UtilsDemo-WX/src/main/java/com/example/entity/RefundOrder.java

@@ -0,0 +1,20 @@
+package com.example.entity;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+import java.io.Serializable;
+
+@ApiModel(value = "申请退款")
+@Data
+public class RefundOrder implements Serializable {
+
+    @ApiModelProperty(value = "订单号")
+    private String orderId;
+
+    @ApiModelProperty(value = "退款原因")
+    private String refundDesc;
+
+
+}

+ 29 - 0
UtilsDemo-WX/src/main/java/com/example/entity/User.java

@@ -0,0 +1,29 @@
+package com.example.entity;
+
+import com.baomidou.mybatisplus.annotations.TableId;
+import com.baomidou.mybatisplus.annotations.TableName;
+import com.baomidou.mybatisplus.enums.IdType;
+import lombok.Data;
+
+import java.util.Date;
+
+@Data
+@TableName(value = "user")
+public class User {
+    @TableId(value = "id", type = IdType.UUID)
+    private String id;
+    private String city;
+    private String country;
+    private String imgUrl;
+    private String nickname;
+    private String openId;
+    private String privileges;
+    private String province;
+    private Integer sex;
+    private Integer type;//0-小程序 1-公众号
+    private Date createDate;
+    private Integer isDel;
+    private String tel;
+    private Date editDate;
+
+}

+ 19 - 0
UtilsDemo-WX/src/main/java/com/example/entity/WxPayFrom.java

@@ -0,0 +1,19 @@
+package com.example.entity;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+@Data
+@ApiModel("微信支付参数")
+public class WxPayFrom {
+    @ApiModelProperty("订单id")
+    private String orderId;
+
+    @ApiModelProperty("微信openid")
+    private String openid;
+
+    @ApiModelProperty("是否开启测试一分钱 test一分钱 空 原价支付")
+    private String tradeType;
+
+}

+ 16 - 0
UtilsDemo-WX/src/main/java/com/example/enums/DelEnum.java

@@ -0,0 +1,16 @@
+package com.example.enums;
+
+public enum DelEnum {
+    NOT_DEL(0, "未删除"), DEL(1, "已删除");
+    private Integer index;
+    private String value;
+
+    DelEnum(Integer index, String value) {
+        this.index = index;
+        this.value = value;
+    }
+
+    public Integer getIndex() {
+        return index;
+    }
+}

+ 29 - 0
UtilsDemo-WX/src/main/java/com/example/enums/PayStatusEnum.java

@@ -0,0 +1,29 @@
+package com.example.enums;
+
+public enum PayStatusEnum {
+
+    INIT(0, "待支付"), SUCCESS(1, "支付成功"), FAIL(2, "支付失败"), APPLY_REFUND(3, "申请退款"),
+    REFUND(4, "已退款"), CLOSED(5, "订单关闭");
+
+    private Integer index;
+    private String value;
+
+
+    PayStatusEnum(Integer status, String value) {
+        this.index = status;
+        this.value = value;
+    }
+
+    public Integer getIndex() {
+        return index;
+    }
+
+    public static String getValue(int index) {
+        for (PayStatusEnum payStatusEnum : PayStatusEnum.values()) {
+            if (payStatusEnum.index == index) {
+                return payStatusEnum.value;
+            }
+        }
+        return "支付失败";
+    }
+}

+ 142 - 0
UtilsDemo-WX/src/main/java/com/example/enums/ResultEnum.java

@@ -0,0 +1,142 @@
+package com.example.enums;
+
+import lombok.Getter;
+
+
+/**
+ * 服务接口返回状态信息枚举定义
+ *
+ * @author yys
+ */
+@Getter
+public enum ResultEnum {
+
+    SUCCESS(0, "操作成功"),
+    FAILURE(1, "操作失败"),
+    FAILED(-1, "非法用户,请输入正确的accessToken"),
+    SYS_ERROR(500, "系统开小差了,请稍后再试"),
+    SYS_BREAK_ERROR(501, "系统服务中断,请稍后再试"),
+    SYS_PASSWD_ERROR(502, "旧密码验证错误,请重试"),
+    PERMISSION_DENIED(403, "没有权限"),
+    ACCOUNT_ALREADY_LOGIN(404, "该账号已经在其他设备上登录"),
+    NOT_LOGIN(401, "请先登录"),
+    RELOGIN(402, "请重新登录"),
+    LOGIN_ACCOUNT_MISS(407, "账号有错误,请检查账号信息"),
+    LOGIN_PWD_MISS(408, "密码有误,请重新输入"),
+    LOGIN_FORBIDED(406, "用户被禁用,请联系管理员"),
+    TOKEN_EXPIRED(405, "登录已过期,请重新登录"),
+    CODE_EXPIRED(409, "验证码不正确,请重新输入"),
+    FAIL_LOGIN_LOGIN_NAME(600, "用户账号或密码不正确"),
+    LOGINING_USER(601, "用户正在登陆系统,不能删除"),
+    ASSAY_APPLIED(602, "请勿重复提交检验申请"),
+    USER_IS_NOT(603, "用户已存在,请登录"),
+    WECHAT_USER_NOT_EXIST(604, "投诉用户不存在"),
+    ENCRYPTION_ERROR(605, "加密操作失败"),
+    PARAMETER_NOT_NULL(606, "参数不能为空"),
+    PARAMETER_INPUT_ERROR(620, "错误的输入参数!"),
+    FILE_NOT_FOUND(607, "文件不存在"),
+    USER_IS_HAVA(608, "用户登陆错误"),
+    WRONG_PHONE(609, "手机号相关错误"),
+    FORM_VALIDATION(610, "表单验证相关错误"),
+    CODE_ERROR(611, "验证码错误或已过期"),
+    FAILED_AUTHORIZE_LOGIN(612, "授权登录失败"),
+    LOGIN_FAILED(613, "登录失败"),
+    USER_IS_NUll(614, "用户数据不存在"),
+    USER_IS_BINDING(615, "该手机号已被使用"),
+    PAYORDER_ADD_ERROR(616, "支付订单新增失败"),
+    PAYORDER_UPDATE_ERROR(617, "支付订单更新失败"),
+    ORG_IS_NOT(618, "企业编码重复!"),
+    CODE_DEL_DUPLICATE(619, "溯源码重复废弃!"),
+    CODE_BATCH_DUPLICATE(621, "产品批次码重复!"),
+    ORDER_PAY_DUPLICATE(622, "订单已支付,请勿重复支付!"),
+    ORDER_PAY_COMPLETE(623, "订单已完成,请勿再次支付!"),
+    GOODS_BUY_AUTH(624, "该用户购买商品超出购买权限!"),
+    WX_AUTH_INFO_SUCCESS(625, "微信认证信息检测成功!"),
+    WX_AUTH_INFO_FAIL(626, "微信认证信息检测失败!"),
+    SIGNATURE_OK(718, "签名成功"),
+    SIGNATURE_ERROR(719, "签名失败"),
+    FILE_UPLOAD_ERROR(720, "文件上传失败!"),
+
+    /** 基础管理错误信息 */
+    ORG_EXIST(800, "部门重复!"),
+
+    /** 店铺管理错误信息 */
+    STORE_EXIST(830,"店铺已存在!"),
+    STORE_LINKPHONE_EXIST(831,"店铺联系电话已存在!"),
+    ROLE_NAME_EXIST(832,"职位重复!"),
+    ROLE_CODE_EXIST(833,"职位编码重复!"),
+    STORE_PERSON_EXIST(834,"职员联系电话已存在!"),
+    STORE_NOT_DEL(835,"职位有关联职员,需要解除以后才能进行删除"),
+
+    WAREHOUSE_NAME_EXIST(837,"仓库名称已存在!"),
+
+    CUSTOMER_CODE_EXIST(840,"客户编码已存在!"),
+    CUSTOMER_LINKPHONE_EXIST(841,"客户联系电话已存在!"),
+
+    PROVIDER_CODE_EXIST(845,"供应商编码已存在!"),
+    PROVIDER_LINKPHONE_EXIST(846,"供应商联系电话已存在!"),
+
+    GOODS_NAME_EXIST(850,"规格型号商品已存在!"),
+    GOODS_REGISTERCODE_EXIST(851,"商品登记证号已存在!"),
+    GOODS_NOT_DEL(852,"商品有交易订单,该商品不能删除"),
+    ORDER_CODE_DUPLICATE(869,"订单编号重复"),
+
+    /** 库存管理错误信息  **/
+    INVENTORYRECORD_STOCKPILEGUID_NULL(860,"库存id不能为空"),
+    INVENTORYRECORD_GOODSGUID_NULL(861,"商品id不能为空"),
+    INVENTORYRECORD_WAREHOUSEGUID_NULL(862,"仓库id不能为空"),
+    INVENTORYRECORD_GOODSNAME_NULL(863,"商品名称不能为空"),
+    INVENTORYRECORD_INITCOUNT_NULL(864,"盘点前库存数量不能为空"),
+    INVENTORYRECORD_INVENTORYCOUNT_NULL(865,"盘存数量不能为空"),
+    INVENTORYRECORD_DIFFCOUNT_NULL(866,"盘存差额不能为空"),
+    INVENTORYRECORD_WAREHOUSENAME_NULL(867,"仓库名称不能为空"),
+    INVENTORYRECORD_REMARK_NULL(868,"备注不能为空"),
+
+
+    /** 参数设置管理模块错误信息  */
+    DICT_TYPE_FOUND(900,"字典类型已经存在"),
+    DICT_CODE_FOUND(901,"字典编码已经存在"),
+    DICT_TYPE_USED(902,"字典类型被下级数据引用,请先删除关联的下级数据"),
+    DICT_VALUE_FOUND(903,"字典值已经存在"),
+    DICT_SEQ_FOUND(904,"字典序列号已经存在"),
+    DICT_NOT_DEL(905,"基础字典类型不能删除"),
+    DICT_CODE_NOTEDIT(906,"基础字典编码不能修改"),
+
+    SELLER_ALREADY_EXIST(720, "该档口已经存在管理员,请选择其他档口"),
+    UNAUTHORIZED(401, "暂未登录或token已经过期");
+
+
+    private Integer code;
+
+    private String msg;
+
+    ResultEnum(Integer code, String msg) {
+        this.code = code;
+        this.msg = msg;
+    }
+
+    /**
+     * 通过枚举code获取枚举msg
+     */
+    public static String getMsgByCode(Integer code) {
+        for (ResultEnum resultEnum : ResultEnum.values()) {
+            if (code.equals(resultEnum.getCode())) {
+                return resultEnum.getMsg();
+            }
+        }
+        return null;
+    }
+
+    /**
+     * 通过枚举code获取枚举对象
+     */
+    public static ResultEnum getResultEnumByCode(Integer code) {
+        for (ResultEnum resultEnum : ResultEnum.values()) {
+            if (code.equals(resultEnum.getCode())) {
+                return resultEnum;
+            }
+        }
+        return null;
+    }
+
+}

+ 16 - 0
UtilsDemo-WX/src/main/java/com/example/enums/SexEnum.java

@@ -0,0 +1,16 @@
+package com.example.enums;
+
+public enum SexEnum {
+    NON(0, "未知"), MAN(1, "男"), WOMEN(2, "女");
+    private Integer index;
+    private String value;
+
+    SexEnum(Integer index, String value) {
+        this.index = index;
+        this.value = value;
+    }
+
+    public Integer getIndex() {
+        return index;
+    }
+}

+ 21 - 0
UtilsDemo-WX/src/main/java/com/example/enums/WX_TypeEnum.java

@@ -0,0 +1,21 @@
+package com.example.enums;
+
+public enum WX_TypeEnum {
+    MI_APP(0, "小程序"), MP(1, "公众号");
+
+    private Integer index;
+    private String value;
+
+    WX_TypeEnum(Integer index, String value) {
+        this.index = index;
+        this.value = value;
+    }
+
+    public Integer getIndex() {
+        return index;
+    }
+
+    public String getValue() {
+        return value;
+    }
+}

+ 17 - 0
UtilsDemo-WX/src/main/java/com/example/exception/ExceptionContoller.java

@@ -0,0 +1,17 @@
+package com.example.exception;
+
+import com.example.base.BaseController;
+import com.example.base.ResponseBase;
+import com.example.enums.ResultEnum;
+import org.springframework.web.bind.annotation.ExceptionHandler;
+import org.springframework.web.bind.annotation.RestControllerAdvice;
+
+@RestControllerAdvice
+public class ExceptionContoller extends BaseController {
+
+    @ExceptionHandler(value = Exception.class)
+    public ResponseBase exception(Exception e) {
+        return responseError(ResultEnum.FAILURE.getCode(), e.getMessage());
+    }
+
+}

+ 65 - 0
UtilsDemo-WX/src/main/java/com/example/filter/AuthGlobalFilter.java

@@ -0,0 +1,65 @@
+package com.example.filter;
+
+import com.example.util.TokenUtil;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.web.servlet.HandlerInterceptor;
+import org.springframework.web.servlet.config.annotation.InterceptorRegistration;
+import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
+import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;
+
+import javax.servlet.http.HttpServletRequest;
+import javax.servlet.http.HttpServletResponse;
+
+
+@Configuration
+@Slf4j
+public class AuthGlobalFilter implements WebMvcConfigurer {
+
+    @Override
+    public void addInterceptors(InterceptorRegistry registry) {
+        InterceptorRegistration registration = registry.addInterceptor(new LoginInterceptors());
+        registration.addPathPatterns("/**");                      //所有路径都被拦截
+        registration.excludePathPatterns(                         //添加不拦截路径
+                "/wechat/h5/authorize",
+                "/wechat/h5/userInfo",
+                "/app/getJsapiTicket",
+                "/wechat/h5/code/{tel}",
+                "/app/payResult",
+                "/app/refundResult",
+                "/**/*.html",            //html静态资源
+                "/swagger-ui.html",
+                "/null/swagger-resources/configuration/ui",
+                "/swagger-resources/configuration/ui",
+                "/swagger-resources/configuration/security",
+                "/swagger-resources/*",
+                "/swagger-resources",
+                "/**/*.js",              //js静态资源
+                "/**/*.css",             //css静态资源
+                "/**/*.png",             //css静态资源
+                "/**/*.woff",
+                "/**/*.woff2",
+                "/error",
+                "/**/*.ttf"
+        );
+    }
+
+
+    public class LoginInterceptors implements HandlerInterceptor {
+        @Override
+        public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
+            String token = request.getHeader("Authorization");
+            System.err.println("获取的token:"+token);
+            String userId = TokenUtil.parseJWT(token).get("uid").toString();
+//            LoginUser loginUser =  tokenService.getLoginUser(request);
+            if(userId == null){
+                log.error("用户未登录,请重新登录");
+//                response.sendRedirect(request.getContextPath()+"http://wx.hw.hongweisoft.com/wxpaydemoh5");
+                return false;
+            }
+            return true;
+        }
+    }
+}
+
+

+ 10 - 0
UtilsDemo-WX/src/main/java/com/example/mapper/UserMapper.java

@@ -0,0 +1,10 @@
+package com.example.mapper;
+
+import com.baomidou.mybatisplus.mapper.BaseMapper;
+import com.example.entity.User;
+import org.apache.ibatis.annotations.Mapper;
+
+@Mapper
+public interface UserMapper extends BaseMapper<User> {
+
+}

+ 231 - 0
UtilsDemo-WX/src/main/java/com/example/redis/RedisCache.java

@@ -0,0 +1,231 @@
+package com.example.redis;
+
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.data.redis.core.BoundSetOperations;
+import org.springframework.data.redis.core.HashOperations;
+import org.springframework.data.redis.core.RedisTemplate;
+import org.springframework.data.redis.core.ValueOperations;
+import org.springframework.stereotype.Component;
+
+import java.util.*;
+import java.util.concurrent.TimeUnit;
+
+/**
+ * spring redis 工具类
+ *
+ * @author ruoyi
+ **/
+@SuppressWarnings(value = { "unchecked", "rawtypes" })
+@Component
+public class RedisCache
+{
+    @Autowired
+    public RedisTemplate redisTemplate;
+
+    /**
+     * 缓存基本的对象,Integer、String、实体类等
+     *
+     * @param key 缓存的键值
+     * @param value 缓存的值
+     */
+    public <T> void setCacheObject(final String key, final T value)
+    {
+        redisTemplate.opsForValue().set(key, value);
+    }
+
+    /**
+     * 缓存基本的对象,Integer、String、实体类等
+     *
+     * @param key 缓存的键值
+     * @param value 缓存的值
+     * @param timeout 时间
+     * @param timeUnit 时间颗粒度
+     */
+    public <T> void setCacheObject(final String key, final T value, final Integer timeout, final TimeUnit timeUnit)
+    {
+        redisTemplate.opsForValue().set(key, value, timeout, timeUnit);
+    }
+
+    /**
+     * 设置有效时间
+     *
+     * @param key Redis键
+     * @param timeout 超时时间
+     * @return true=设置成功;false=设置失败
+     */
+    public boolean expire(final String key, final long timeout)
+    {
+        return expire(key, timeout, TimeUnit.SECONDS);
+    }
+
+    /**
+     * 设置有效时间
+     *
+     * @param key Redis键
+     * @param timeout 超时时间
+     * @param unit 时间单位
+     * @return true=设置成功;false=设置失败
+     */
+    public boolean expire(final String key, final long timeout, final TimeUnit unit)
+    {
+        return redisTemplate.expire(key, timeout, unit);
+    }
+
+    /**
+     * 获得缓存的基本对象。
+     *
+     * @param key 缓存键值
+     * @return 缓存键值对应的数据
+     */
+    public <T> T getCacheObject(final String key)
+    {
+        ValueOperations<String, T> operation = redisTemplate.opsForValue();
+        return operation.get(key);
+    }
+
+    /**
+     * 删除单个对象
+     *
+     * @param key
+     */
+    public boolean deleteObject(final String key)
+    {
+        return redisTemplate.delete(key);
+    }
+
+    /**
+     * 删除集合对象
+     *
+     * @param collection 多个对象
+     * @return
+     */
+    public long deleteObject(final Collection collection)
+    {
+        return redisTemplate.delete(collection);
+    }
+
+    /**
+     * 缓存List数据
+     *
+     * @param key 缓存的键值
+     * @param dataList 待缓存的List数据
+     * @return 缓存的对象
+     */
+    public <T> long setCacheList(final String key, final List<T> dataList)
+    {
+        Long count = redisTemplate.opsForList().rightPushAll(key, dataList);
+        return count == null ? 0 : count;
+    }
+
+    /**
+     * 获得缓存的list对象
+     *
+     * @param key 缓存的键值
+     * @return 缓存键值对应的数据
+     */
+    public <T> List<T> getCacheList(final String key)
+    {
+        return redisTemplate.opsForList().range(key, 0, -1);
+    }
+
+    /**
+     * 缓存Set
+     *
+     * @param key 缓存键值
+     * @param dataSet 缓存的数据
+     * @return 缓存数据的对象
+     */
+    public <T> BoundSetOperations<String, T> setCacheSet(final String key, final Set<T> dataSet)
+    {
+        BoundSetOperations<String, T> setOperation = redisTemplate.boundSetOps(key);
+        Iterator<T> it = dataSet.iterator();
+        while (it.hasNext())
+        {
+            setOperation.add(it.next());
+        }
+        return setOperation;
+    }
+
+    /**
+     * 获得缓存的set
+     *
+     * @param key
+     * @return
+     */
+    public <T> Set<T> getCacheSet(final String key)
+    {
+        return redisTemplate.opsForSet().members(key);
+    }
+
+    /**
+     * 缓存Map
+     *
+     * @param key
+     * @param dataMap
+     */
+    public <T> void setCacheMap(final String key, final Map<String, T> dataMap)
+    {
+        if (dataMap != null) {
+            redisTemplate.opsForHash().putAll(key, dataMap);
+        }
+    }
+
+    /**
+     * 获得缓存的Map
+     *
+     * @param key
+     * @return
+     */
+    public <T> Map<String, T> getCacheMap(final String key)
+    {
+        return redisTemplate.opsForHash().entries(key);
+    }
+
+    /**
+     * 往Hash中存入数据
+     *
+     * @param key Redis键
+     * @param hKey Hash键
+     * @param value 值
+     */
+    public <T> void setCacheMapValue(final String key, final String hKey, final T value)
+    {
+        redisTemplate.opsForHash().put(key, hKey, value);
+    }
+
+    /**
+     * 获取Hash中的数据
+     *
+     * @param key Redis键
+     * @param hKey Hash键
+     * @return Hash中的对象
+     */
+    public <T> T getCacheMapValue(final String key, final String hKey)
+    {
+        HashOperations<String, String, T> opsForHash = redisTemplate.opsForHash();
+        return opsForHash.get(key, hKey);
+    }
+
+    /**
+     * 获取多个Hash中的数据
+     *
+     * @param key Redis键
+     * @param hKeys Hash键集合
+     * @return Hash对象集合
+     */
+    public <T> List<T> getMultiCacheMapValue(final String key, final Collection<Object> hKeys)
+    {
+        return redisTemplate.opsForHash().multiGet(key, hKeys);
+    }
+
+    /**
+     * 获得缓存的基本对象列表
+     *
+     * @param pattern 字符串前缀
+     * @return 对象列表
+     */
+    public Collection<String> keys(final String pattern)
+    {
+        return redisTemplate.keys(pattern);
+    }
+}

+ 14 - 0
UtilsDemo-WX/src/main/java/com/example/service/UserService.java

@@ -0,0 +1,14 @@
+package com.example.service;
+
+import com.baomidou.mybatisplus.service.IService;
+import com.example.entity.User;
+import com.example.vo.UserUpdateVO;
+import me.chanjar.weixin.mp.bean.result.WxMpUser;
+
+public interface UserService extends IService<User> {
+    void addUser(WxMpUser wxMpUser);
+
+    User getUserByOpenId(String openId);
+
+    void updateUser(String openId, UserUpdateVO updateVO);
+}

+ 53 - 0
UtilsDemo-WX/src/main/java/com/example/service/impl/UserServiceImpl.java

@@ -0,0 +1,53 @@
+package com.example.service.impl;
+
+import com.baomidou.mybatisplus.service.impl.ServiceImpl;
+import com.example.entity.User;
+import com.example.enums.DelEnum;
+import com.example.enums.WX_TypeEnum;
+import com.example.mapper.UserMapper;
+import com.example.service.UserService;
+import com.example.vo.UserUpdateVO;
+import me.chanjar.weixin.mp.bean.result.WxMpUser;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+import java.util.Date;
+
+@Service
+public class UserServiceImpl extends ServiceImpl<UserMapper,User> implements UserService {
+
+    @Autowired
+    private UserMapper userMapper;
+
+    @Override
+    public void addUser(WxMpUser wxMpUser) {
+        User user = new User();
+        user.setCity(wxMpUser.getCity());
+        user.setCountry(wxMpUser.getCountry());
+        user.setProvince(wxMpUser.getProvince());
+        user.setCreateDate(new Date());
+        user.setImgUrl(wxMpUser.getHeadImgUrl());
+        user.setNickname(wxMpUser.getNickname());
+        user.setOpenId(wxMpUser.getOpenId());
+        user.setSex(wxMpUser.getSex());
+        user.setPrivileges(wxMpUser.getPrivileges().toString());
+        user.setIsDel(DelEnum.NOT_DEL.getIndex());
+        user.setType(WX_TypeEnum.MP.getIndex());
+        userMapper.insert(user);
+    }
+
+    @Override
+    public User getUserByOpenId(String openId) {
+        User user = new User();
+        user.setOpenId(openId);
+        User result = userMapper.selectOne(user);
+        return result;
+    }
+
+    @Override
+    public void updateUser(String openId, UserUpdateVO updateVO) {
+        User user = getUserByOpenId(openId);
+        user.setTel(updateVO.getPhone());
+        userMapper.updateAllColumnById(user);
+    }
+}

+ 84 - 0
UtilsDemo-WX/src/main/java/com/example/util/BeanCopyUtil.java

@@ -0,0 +1,84 @@
+package com.example.util;
+
+import com.google.common.collect.Lists;
+import org.dozer.DozerBeanMapper;
+import org.springframework.beans.BeanUtils;
+import org.springframework.beans.BeanWrapper;
+import org.springframework.beans.BeanWrapperImpl;
+
+import java.beans.PropertyDescriptor;
+import java.util.Collection;
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+
+/**
+ * Java对象拷贝、转换工具类
+ *
+ * @author yys
+ */
+public final class BeanCopyUtil {
+
+    private BeanCopyUtil() {
+    }
+
+    private static DozerBeanMapper dozer = new DozerBeanMapper();
+
+    /**
+     * 源对象转换为目标对象
+     *
+     * @param source           源对象
+     * @param destinationClass 目标Class对象
+     * @return 目标对象
+     */
+    public static <T> T convertBean(Object source, Class<T> destinationClass) {
+        return dozer.map(source, destinationClass);
+    }
+
+    /**
+     * 源List中对象转换为目标对象
+     *
+     * @param sourceList       源对象List
+     * @param destinationClass 目标List中的Class对象
+     * @return 转换后的List集合
+     */
+    public static <T> List<T> convertList(Collection<?> sourceList, Class<T> destinationClass) {
+        List<T> destinationList = Lists.newArrayList();
+        for (Object sourceObject : sourceList) {
+            T destinationObject = dozer.map(sourceObject, destinationClass);
+            destinationList.add(destinationObject);
+        }
+        return destinationList;
+    }
+
+    /**
+     * 将源对象的属性复制目标对象
+     *
+     * @param source            源对象
+     * @param destinationObject 目标对象
+     */
+    public static void copyBean(Object source, Object destinationObject) {
+        dozer.map(source, destinationObject);
+    }
+
+    /**
+     * 拷贝不为空的属性覆盖目标对象(常用于查询更新场景)
+     *
+     * @param src    源对象
+     * @param target 目标对象
+     */
+    public static void copyNotNullProperties(Object src, Object target) {
+        BeanWrapper srcBean = new BeanWrapperImpl(src);
+        PropertyDescriptor[] pds = srcBean.getPropertyDescriptors();
+        Set<String> emptyName = new HashSet<String>();
+        for (PropertyDescriptor p : pds) {
+            Object srcValue = srcBean.getPropertyValue(p.getName());
+            if (srcValue == null) {
+                emptyName.add(p.getName());
+            }
+        }
+        String[] result = new String[emptyName.size()];
+        BeanUtils.copyProperties(src, target, emptyName.toArray(result));
+    }
+
+}

+ 155 - 0
UtilsDemo-WX/src/main/java/com/example/util/EscapeUtil.java

@@ -0,0 +1,155 @@
+package com.example.util;
+
+import org.bouncycastle.i18n.filter.HTMLFilter;
+
+/**
+ * 转义和反转义工具类
+ * 
+ * @author ruoyi
+ */
+public class EscapeUtil
+{
+    public static final String RE_HTML_MARK = "(<[^<]*?>)|(<[\\s]*?/[^<]*?>)|(<[^<]*?/[\\s]*?>)";
+
+    private static final char[][] TEXT = new char[64][];
+
+    static
+    {
+        for (int i = 0; i < 64; i++)
+        {
+            TEXT[i] = new char[] { (char) i };
+        }
+
+        // special HTML characters
+        TEXT['\''] = "&#039;".toCharArray(); // 单引号
+        TEXT['"'] = "&#34;".toCharArray(); // 单引号
+        TEXT['&'] = "&#38;".toCharArray(); // &符
+        TEXT['<'] = "&#60;".toCharArray(); // 小于号
+        TEXT['>'] = "&#62;".toCharArray(); // 大于号
+    }
+
+    /**
+     * 转义文本中的HTML字符为安全的字符
+     * 
+     * @param text 被转义的文本
+     * @return 转义后的文本
+     */
+    public static String escape(String text)
+    {
+        return encode(text);
+    }
+
+    /**
+     * 还原被转义的HTML特殊字符
+     * 
+     * @param content 包含转义符的HTML内容
+     * @return 转换后的字符串
+     */
+    public static String unescape(String content)
+    {
+        return decode(content);
+    }
+
+    /**
+     * 清除所有HTML标签,但是不删除标签内的内容
+     * 
+     * @param content 文本
+     * @return 清除标签后的文本
+     */
+//    public static String clean(String content)
+//    {
+//        return new HTMLFilter().wait(content);
+//    }
+
+    /**
+     * Escape编码
+     * 
+     * @param text 被编码的文本
+     * @return 编码后的字符
+     */
+    private static String encode(String text)
+    {
+        int len;
+        if ((text == null) || ((len = text.length()) == 0))
+        {
+            return StringUtils.EMPTY;
+        }
+        StringBuilder buffer = new StringBuilder(len + (len >> 2));
+        char c;
+        for (int i = 0; i < len; i++)
+        {
+            c = text.charAt(i);
+            if (c < 64)
+            {
+                buffer.append(TEXT[c]);
+            }
+            else
+            {
+                buffer.append(c);
+            }
+        }
+        return buffer.toString();
+    }
+
+    /**
+     * Escape解码
+     * 
+     * @param content 被转义的内容
+     * @return 解码后的字符串
+     */
+    public static String decode(String content)
+    {
+        if (StringUtils.isEmpty(content))
+        {
+            return content;
+        }
+
+        StringBuilder tmp = new StringBuilder(content.length());
+        int lastPos = 0, pos = 0;
+        char ch;
+        while (lastPos < content.length())
+        {
+            pos = content.indexOf("%", lastPos);
+            if (pos == lastPos)
+            {
+                if (content.charAt(pos + 1) == 'u')
+                {
+                    ch = (char) Integer.parseInt(content.substring(pos + 2, pos + 6), 16);
+                    tmp.append(ch);
+                    lastPos = pos + 6;
+                }
+                else
+                {
+                    ch = (char) Integer.parseInt(content.substring(pos + 1, pos + 3), 16);
+                    tmp.append(ch);
+                    lastPos = pos + 3;
+                }
+            }
+            else
+            {
+                if (pos == -1)
+                {
+                    tmp.append(content.substring(lastPos));
+                    lastPos = content.length();
+                }
+                else
+                {
+                    tmp.append(content.substring(lastPos, pos));
+                    lastPos = pos;
+                }
+            }
+        }
+        return tmp.toString();
+    }
+
+    public static void main(String[] args)
+    {
+        String html = "<script>alert(1);</script>";
+        // String html = "<scr<script>ipt>alert(\"XSS\")</scr<script>ipt>";
+        // String html = "<123";
+        // String html = "123>";
+//        System.out.println(EscapeUtil.clean(html));
+        System.out.println(EscapeUtil.escape(html));
+        System.out.println(EscapeUtil.unescape(html));
+    }
+}

+ 129 - 0
UtilsDemo-WX/src/main/java/com/example/util/GeneratorIdUtils.java

@@ -0,0 +1,129 @@
+package com.example.util;
+
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+import java.text.SimpleDateFormat;
+
+/**
+ * 订购业务唯一订单号实现
+ *
+ * @author ouyangjun
+ */
+public class GeneratorIdUtils {
+
+    /**
+     * 用ip地址最后几个字节标示
+     */
+    private long workerId;
+    /**
+     * 可配置在properties中,启动时加载,此处默认先写成0
+     */
+    private long datacenterId = 0L;
+
+    private long sequence = 0L;
+    /**
+     * 节点ID长度
+     */
+    private long workerIdBits = 8L;
+    /**
+     * 序列号12位
+     */
+    private long sequenceBits = 12L;
+    /**
+     * 机器节点左移12位
+     */
+    private long workerIdShift = sequenceBits;
+    /**
+     * 数据中心节点左移17位
+     */
+    private long datacenterIdShift = sequenceBits + workerIdBits;
+    /**
+     * 4095
+     */
+    private long sequenceMask = -1L ^ (-1L << sequenceBits);
+
+    private long lastTimestamp = -1L;
+
+    public GeneratorIdUtils() {
+        workerId = 0x000000FF & getLastIP();
+    }
+
+    /**
+     * 调用该方法,获取序列ID
+     *
+     * @return
+     */
+    public synchronized String nextId() {
+        //获取当前毫秒数
+        long timestamp = currentTime();
+        //如果服务器时间有问题(时钟后退) 报错。
+        if (timestamp < lastTimestamp) {
+            throw new RuntimeException(String.format("时钟向后移动。拒绝在%d毫秒内生成id", lastTimestamp - timestamp));
+        }
+        //如果上次生成时间和当前时间相同,在同一毫秒内
+        if (lastTimestamp == timestamp) {
+            //sequence自增,因为sequence只有12bit,所以和sequenceMask相与一下,去掉高位
+            sequence = (sequence + 1) & sequenceMask;
+            //判断是否溢出,也就是每毫秒内超过4095,当为4096时,与sequenceMask相与,sequence就等于0
+            if (sequence == 0) {
+                //自旋等待到下一毫秒
+                timestamp = nextMillis(lastTimestamp);
+            }
+        } else {
+            //如果和上次生成时间不同,重置sequence,就是下一毫秒开始,sequence计数重新从0开始累加
+            sequence = 0L;
+        }
+        lastTimestamp = timestamp;
+
+        long suffix = (datacenterId << datacenterIdShift) | (workerId << workerIdShift) | sequence;
+
+        // 格式化日期
+        SimpleDateFormat timePe = new SimpleDateFormat("yyyyMMddHHMMssSSS");
+        String datePrefix = timePe.format(timestamp);
+
+        return datePrefix + suffix;
+    }
+
+    /**
+     * 比较当前时间戳和下一个时间戳,如果下一个时间戳等于或小于当前时间戳,则循环获取下个时间戳
+     * 该方法主要是避免同一时间获取同一时间戳
+     *
+     * @param lastTimestamp
+     * @return
+     */
+    protected long nextMillis(long lastTimestamp) {
+        long timestamp = currentTime();
+        while (timestamp <= lastTimestamp) {
+            timestamp = currentTime();
+        }
+        return timestamp;
+    }
+
+    /**
+     * 获取系统当前时间戳
+     *
+     * @return
+     */
+    protected long currentTime() {
+        return System.currentTimeMillis();
+    }
+
+    /**
+     * 获取当前本地IP
+     *
+     * @return
+     */
+    private byte getLastIP() {
+        byte lastip = 0;
+        try {
+            InetAddress ip = InetAddress.getLocalHost();
+            byte[] ipByte = ip.getAddress();
+            lastip = ipByte[ipByte.length - 1];
+        } catch (UnknownHostException e) {
+            e.printStackTrace();
+        }
+        return lastip;
+    }
+
+
+}

+ 45 - 0
UtilsDemo-WX/src/main/java/com/example/util/IdUtils.java

@@ -0,0 +1,45 @@
+package com.example.util;
+
+
+/**
+ * ID生成器工具类
+ *
+ * @author ruoyi
+ */
+public class IdUtils {
+    /**
+     * 获取随机UUID
+     *
+     * @return 随机UUID
+     */
+    public static String randomUUID() {
+        return UUID.randomUUID().toString();
+    }
+
+    /**
+     * 简化的UUID,去掉了横线
+     *
+     * @return 简化的UUID,去掉了横线
+     */
+    public static String simpleUUID() {
+        return UUID.randomUUID().toString(true);
+    }
+
+    /**
+     * 获取随机UUID,使用性能更好的ThreadLocalRandom生成UUID
+     *
+     * @return 随机UUID
+     */
+    public static String fastUUID() {
+        return UUID.fastUUID().toString();
+    }
+
+    /**
+     * 简化的UUID,去掉了横线,使用性能更好的ThreadLocalRandom生成UUID
+     *
+     * @return 简化的UUID,去掉了横线
+     */
+    public static String fastSimpleUUID() {
+        return UUID.fastUUID().toString(true);
+    }
+}

+ 194 - 0
UtilsDemo-WX/src/main/java/com/example/util/IpUtils.java

@@ -0,0 +1,194 @@
+package com.example.util;
+
+
+import javax.servlet.http.HttpServletRequest;
+import java.net.InetAddress;
+import java.net.UnknownHostException;
+
+/**
+ * 获取IP方法
+ * 
+ * @author ruoyi
+ */
+public class IpUtils
+{
+    public static String getIpAddr(HttpServletRequest request)
+    {
+        if (request == null)
+        {
+            return "unknown";
+        }
+        String ip = request.getHeader("x-forwarded-for");
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
+        {
+            ip = request.getHeader("Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
+        {
+            ip = request.getHeader("X-Forwarded-For");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
+        {
+            ip = request.getHeader("WL-Proxy-Client-IP");
+        }
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
+        {
+            ip = request.getHeader("X-Real-IP");
+        }
+
+        if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip))
+        {
+            ip = request.getRemoteAddr();
+        }
+        return "0:0:0:0:0:0:0:1".equals(ip) ? "127.0.0.1" : "0:0:0:0:0:0:0:1";
+    }
+
+    public static boolean internalIp(String ip)
+    {
+        byte[] addr = textToNumericFormatV4(ip);
+        return internalIp(addr) || "127.0.0.1".equals(ip);
+    }
+
+    private static boolean internalIp(byte[] addr)
+    {
+        if (StringUtils.isNull(addr) || addr.length < 2)
+        {
+            return true;
+        }
+        final byte b0 = addr[0];
+        final byte b1 = addr[1];
+        // 10.x.x.x/8
+        final byte SECTION_1 = 0x0A;
+        // 172.16.x.x/12
+        final byte SECTION_2 = (byte) 0xAC;
+        final byte SECTION_3 = (byte) 0x10;
+        final byte SECTION_4 = (byte) 0x1F;
+        // 192.168.x.x/16
+        final byte SECTION_5 = (byte) 0xC0;
+        final byte SECTION_6 = (byte) 0xA8;
+        switch (b0)
+        {
+            case SECTION_1:
+                return true;
+            case SECTION_2:
+                if (b1 >= SECTION_3 && b1 <= SECTION_4)
+                {
+                    return true;
+                }
+            case SECTION_5:
+                switch (b1)
+                {
+                    case SECTION_6:
+                        return true;
+                }
+            default:
+                return false;
+        }
+    }
+
+    /**
+     * 将IPv4地址转换成字节
+     * 
+     * @param text IPv4地址
+     * @return byte 字节
+     */
+    public static byte[] textToNumericFormatV4(String text)
+    {
+        if (text.length() == 0)
+        {
+            return null;
+        }
+
+        byte[] bytes = new byte[4];
+        String[] elements = text.split("\\.", -1);
+        try
+        {
+            long l;
+            int i;
+            switch (elements.length)
+            {
+                case 1:
+                    l = Long.parseLong(elements[0]);
+                    if ((l < 0L) || (l > 4294967295L)) {
+                        return null;
+                    }
+                    bytes[0] = (byte) (int) (l >> 24 & 0xFF);
+                    bytes[1] = (byte) (int) ((l & 0xFFFFFF) >> 16 & 0xFF);
+                    bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF);
+                    bytes[3] = (byte) (int) (l & 0xFF);
+                    break;
+                case 2:
+                    l = Integer.parseInt(elements[0]);
+                    if ((l < 0L) || (l > 255L)) {
+                        return null;
+                    }
+                    bytes[0] = (byte) (int) (l & 0xFF);
+                    l = Integer.parseInt(elements[1]);
+                    if ((l < 0L) || (l > 16777215L)) {
+                        return null;
+                    }
+                    bytes[1] = (byte) (int) (l >> 16 & 0xFF);
+                    bytes[2] = (byte) (int) ((l & 0xFFFF) >> 8 & 0xFF);
+                    bytes[3] = (byte) (int) (l & 0xFF);
+                    break;
+                case 3:
+                    for (i = 0; i < 2; ++i)
+                    {
+                        l = Integer.parseInt(elements[i]);
+                        if ((l < 0L) || (l > 255L)) {
+                            return null;
+                        }
+                        bytes[i] = (byte) (int) (l & 0xFF);
+                    }
+                    l = Integer.parseInt(elements[2]);
+                    if ((l < 0L) || (l > 65535L)) {
+                        return null;
+                    }
+                    bytes[2] = (byte) (int) (l >> 8 & 0xFF);
+                    bytes[3] = (byte) (int) (l & 0xFF);
+                    break;
+                case 4:
+                    for (i = 0; i < 4; ++i)
+                    {
+                        l = Integer.parseInt(elements[i]);
+                        if ((l < 0L) || (l > 255L)) {
+                            return null;
+                        }
+                        bytes[i] = (byte) (int) (l & 0xFF);
+                    }
+                    break;
+                default:
+                    return null;
+            }
+        }
+        catch (NumberFormatException e)
+        {
+            return null;
+        }
+        return bytes;
+    }
+
+    public static String getHostIp()
+    {
+        try
+        {
+            return InetAddress.getLocalHost().getHostAddress();
+        }
+        catch (UnknownHostException e)
+        {
+        }
+        return "127.0.0.1";
+    }
+
+    public static String getHostName()
+    {
+        try
+        {
+            return InetAddress.getLocalHost().getHostName();
+        }
+        catch (UnknownHostException e)
+        {
+        }
+        return "未知";
+    }
+}

+ 880 - 0
UtilsDemo-WX/src/main/java/com/example/util/RedisUtil.java

@@ -0,0 +1,880 @@
+package com.example.util;
+
+
+import java.util.List;
+
+import java.util.Map;
+
+import java.util.Set;
+
+import java.util.concurrent.TimeUnit;
+
+
+import org.springframework.beans.factory.annotation.Autowired;
+
+import org.springframework.data.redis.core.RedisTemplate;
+
+import org.springframework.stereotype.Component;
+
+import org.springframework.util.CollectionUtils;
+
+
+/**
+ * Redis工具类
+ */
+
+@Component
+public final class RedisUtil {
+
+
+    @Autowired
+    private RedisTemplate<String, Object> redisTemplate;
+
+    // =============================common============================
+
+    /**
+     * 指定缓存失效时间
+     *
+     * @param key  键
+     * @param time 时间(秒)
+     * @return
+     */
+
+    public boolean expire(String key, long time) {
+
+        try {
+
+            if (time > 0) {
+
+                redisTemplate.expire(key, time, TimeUnit.SECONDS);
+
+            }
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 根据key 获取过期时间
+     *
+     * @param key 键 不能为null
+     * @return 时间(秒) 返回0代表为永久有效
+     */
+
+    public long getExpire(String key) {
+
+        return redisTemplate.getExpire(key, TimeUnit.SECONDS);
+
+    }
+
+
+    /**
+     * 判断key是否存在
+     *
+     * @param key 键
+     * @return true 存在 false不存在
+     */
+
+    public boolean hasKey(String key) {
+
+        try {
+
+            return redisTemplate.hasKey(key);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 删除缓存
+     *
+     * @param key 可以传一个值 或多个
+     */
+
+    @SuppressWarnings("unchecked")
+
+    public void del(String... key) {
+
+        if (key != null && key.length > 0) {
+
+            if (key.length == 1) {
+
+                redisTemplate.delete(key[0]);
+
+            } else {
+
+                redisTemplate.delete(CollectionUtils.arrayToList(key));
+
+            }
+
+        }
+
+    }
+
+
+    // ============================String=============================
+
+    /**
+     * 普通缓存获取
+     *
+     * @param key 键
+     * @return 值
+     */
+
+    public Object get(String key) {
+
+        return key == null ? null : redisTemplate.opsForValue().get(key);
+
+    }
+
+
+    /**
+     * 普通缓存放入
+     *
+     * @param key   键
+     * @param value 值
+     * @return true成功 false失败
+     */
+
+    public boolean set(String key, Object value) {
+
+        try {
+
+            redisTemplate.opsForValue().set(key, value);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+
+    }
+
+
+    /**
+     * 普通缓存放入并设置时间
+     *
+     * @param key   键
+     * @param value 值
+     * @param time  时间(秒) time要大于0 如果time小于等于0 将设置无限期
+     * @return true成功 false 失败
+     */
+
+    public boolean set(String key, Object value, long time) {
+
+        try {
+
+            if (time > 0) {
+
+                redisTemplate.opsForValue().set(key, value, time, TimeUnit.SECONDS);
+
+            } else {
+
+                set(key, value);
+
+            }
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 递增
+     *
+     * @param key   键
+     * @param delta 要增加几(大于0)
+     * @return
+     */
+
+    public long incr(String key, long delta) {
+
+        if (delta < 0) {
+
+            throw new RuntimeException("递增因子必须大于0");
+
+        }
+
+        return redisTemplate.opsForValue().increment(key, delta);
+
+    }
+
+
+    /**
+     * 递减
+     *
+     * @param key   键
+     * @param delta 要减少几(小于0)
+     * @return
+     */
+
+    public long decr(String key, long delta) {
+
+        if (delta < 0) {
+
+            throw new RuntimeException("递减因子必须大于0");
+
+        }
+
+        return redisTemplate.opsForValue().increment(key, -delta);
+
+    }
+
+
+    // ================================Map=================================
+
+    /**
+     * HashGet
+     *
+     * @param key  键 不能为null
+     * @param item 项 不能为null
+     * @return 值
+     */
+
+    public Object hget(String key, String item) {
+
+        return redisTemplate.opsForHash().get(key, item);
+
+    }
+
+
+    /**
+     * 获取hashKey对应的所有键值
+     *
+     * @param key 键
+     * @return 对应的多个键值
+     */
+
+    public Map<Object, Object> hmget(String key) {
+
+        return redisTemplate.opsForHash().entries(key);
+
+    }
+
+
+    /**
+     * HashSet
+     *
+     * @param key 键
+     * @param map 对应多个键值
+     * @return true 成功 false 失败
+     */
+
+    public boolean hmset(String key, Map<String, Object> map) {
+
+        try {
+
+            redisTemplate.opsForHash().putAll(key, map);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * HashSet 并设置时间
+     *
+     * @param key  键
+     * @param map  对应多个键值
+     * @param time 时间(秒)
+     * @return true成功 false失败
+     */
+
+    public boolean hmset(String key, Map<String, Object> map, long time) {
+
+        try {
+
+            redisTemplate.opsForHash().putAll(key, map);
+
+            if (time > 0) {
+
+                expire(key, time);
+
+            }
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 向一张hash表中放入数据,如果不存在将创建
+     *
+     * @param key   键
+     * @param item  项
+     * @param value 值
+     * @return true 成功 false失败
+     */
+
+    public boolean hset(String key, String item, Object value) {
+
+        try {
+
+            redisTemplate.opsForHash().put(key, item, value);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 向一张hash表中放入数据,如果不存在将创建
+     *
+     * @param key   键
+     * @param item  项
+     * @param value 值
+     * @param time  时间(秒) 注意:如果已存在的hash表有时间,这里将会替换原有的时间
+     * @return true 成功 false失败
+     */
+
+    public boolean hset(String key, String item, Object value, long time) {
+
+        try {
+
+            redisTemplate.opsForHash().put(key, item, value);
+
+            if (time > 0) {
+
+                expire(key, time);
+
+            }
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 删除hash表中的值
+     *
+     * @param key  键 不能为null
+     * @param item 项 可以使多个 不能为null
+     */
+
+    public void hdel(String key, Object... item) {
+
+        redisTemplate.opsForHash().delete(key, item);
+
+    }
+
+
+    /**
+     * 判断hash表中是否有该项的值
+     *
+     * @param key  键 不能为null
+     * @param item 项 不能为null
+     * @return true 存在 false不存在
+     */
+
+    public boolean hHasKey(String key, String item) {
+
+        return redisTemplate.opsForHash().hasKey(key, item);
+
+    }
+
+
+    /**
+     * hash递增 如果不存在,就会创建一个 并把新增后的值返回
+     *
+     * @param key  键
+     * @param item 项
+     * @param by   要增加几(大于0)
+     * @return
+     */
+
+    public double hincr(String key, String item, double by) {
+
+        return redisTemplate.opsForHash().increment(key, item, by);
+
+    }
+
+
+    /**
+     * hash递减
+     *
+     * @param key  键
+     * @param item 项
+     * @param by   要减少记(小于0)
+     * @return
+     */
+
+    public double hdecr(String key, String item, double by) {
+
+        return redisTemplate.opsForHash().increment(key, item, -by);
+
+    }
+
+
+    // ============================set=============================
+
+    /**
+     * 根据key获取Set中的所有值
+     *
+     * @param key 键
+     * @return
+     */
+
+    public Set<Object> sGet(String key) {
+
+        try {
+
+            return redisTemplate.opsForSet().members(key);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return null;
+
+        }
+
+    }
+
+
+    /**
+     * 根据value从一个set中查询,是否存在
+     *
+     * @param key   键
+     * @param value 值
+     * @return true 存在 false不存在
+     */
+
+    public boolean sHasKey(String key, Object value) {
+
+        try {
+
+            return redisTemplate.opsForSet().isMember(key, value);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 将数据放入set缓存
+     *
+     * @param key    键
+     * @param values 值 可以是多个
+     * @return 成功个数
+     */
+
+    public long sSet(String key, Object... values) {
+
+        try {
+
+            return redisTemplate.opsForSet().add(key, values);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+
+    /**
+     * 将set数据放入缓存
+     *
+     * @param key    键
+     * @param time   时间(秒)
+     * @param values 值 可以是多个
+     * @return 成功个数
+     */
+
+    public long sSetAndTime(String key, long time, Object... values) {
+
+        try {
+
+            Long count = redisTemplate.opsForSet().add(key, values);
+
+            if (time > 0)
+
+                expire(key, time);
+
+            return count;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+
+    /**
+     * 获取set缓存的长度
+     *
+     * @param key 键
+     * @return
+     */
+
+    public long sGetSetSize(String key) {
+
+        try {
+
+            return redisTemplate.opsForSet().size(key);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+
+    /**
+     * 移除值为value的
+     *
+     * @param key    键
+     * @param values 值 可以是多个
+     * @return 移除的个数
+     */
+
+    public long setRemove(String key, Object... values) {
+
+        try {
+
+            Long count = redisTemplate.opsForSet().remove(key, values);
+
+            return count;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+    // ===============================list=================================
+
+
+    /**
+     * 获取list缓存的内容
+     *
+     * @param key   键
+     * @param start 开始
+     * @param end   结束 0 到 -1代表所有值
+     * @return
+     */
+
+    public List<Object> lGet(String key, long start, long end) {
+
+        try {
+
+            return redisTemplate.opsForList().range(key, start, end);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return null;
+
+        }
+
+    }
+
+
+    /**
+     * 获取list缓存的长度
+     *
+     * @param key 键
+     * @return
+     */
+
+    public long lGetListSize(String key) {
+
+        try {
+
+            return redisTemplate.opsForList().size(key);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+
+    /**
+     * 通过索引 获取list中的值
+     *
+     * @param key   键
+     * @param index 索引 index>=0时, 0 表头,1 第二个元素,依次类推;index<0时,-1,表尾,-2倒数第二个元素,依次类推
+     * @return
+     */
+
+    public Object lGetIndex(String key, long index) {
+
+        try {
+
+            return redisTemplate.opsForList().index(key, index);
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return null;
+
+        }
+
+    }
+
+
+    /**
+     * 将list放入缓存
+     *
+     * @param key   键
+     * @param value 值
+     * @param time  时间(秒)
+     * @return
+     */
+
+    public boolean lSet(String key, Object value) {
+
+        try {
+
+            redisTemplate.opsForList().rightPush(key, value);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 将list放入缓存
+     *
+     * @param key   键
+     * @param value 值
+     * @param time  时间(秒)
+     * @return
+     */
+
+    public boolean lSet(String key, Object value, long time) {
+
+        try {
+
+            redisTemplate.opsForList().rightPush(key, value);
+
+            if (time > 0)
+
+                expire(key, time);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 将list放入缓存
+     *
+     * @param key   键
+     * @param value 值
+     * @param time  时间(秒)
+     * @return
+     */
+
+    public boolean lSet(String key, List<Object> value) {
+
+        try {
+
+            redisTemplate.opsForList().rightPushAll(key, value);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 将list放入缓存
+     *
+     * @param key   键
+     * @param value 值
+     * @param time  时间(秒)
+     * @return
+     */
+
+    public boolean lSet(String key, List<Object> value, long time) {
+
+        try {
+
+            redisTemplate.opsForList().rightPushAll(key, value);
+
+            if (time > 0)
+
+                expire(key, time);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 根据索引修改list中的某条数据
+     *
+     * @param key   键
+     * @param index 索引
+     * @param value 值
+     * @return
+     */
+
+    public boolean lUpdateIndex(String key, long index, Object value) {
+
+        try {
+
+            redisTemplate.opsForList().set(key, index, value);
+
+            return true;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return false;
+
+        }
+
+    }
+
+
+    /**
+     * 移除N个值为value
+     *
+     * @param key   键
+     * @param count 移除多少个
+     * @param value 值
+     * @return 移除的个数
+     */
+
+    public long lRemove(String key, long count, Object value) {
+
+        try {
+
+            Long remove = redisTemplate.opsForList().remove(key, count, value);
+
+            return remove;
+
+        } catch (Exception e) {
+
+            e.printStackTrace();
+
+            return 0;
+
+        }
+
+    }
+
+}

+ 454 - 0
UtilsDemo-WX/src/main/java/com/example/util/StringUtils.java

@@ -0,0 +1,454 @@
+package com.example.util;
+
+
+import java.util.*;
+
+/**
+ * 字符串工具类
+ * 
+ * @author ruoyi
+ */
+public class StringUtils extends org.apache.commons.lang3.StringUtils
+{
+    /** 空字符串 */
+    private static final String NULLSTR = "";
+
+    /** 下划线 */
+    private static final char SEPARATOR = '_';
+
+    /**
+     * 获取参数不为空值
+     * 
+     * @param value defaultValue 要判断的value
+     * @return value 返回值
+     */
+    public static <T> T nvl(T value, T defaultValue)
+    {
+        return value != null ? value : defaultValue;
+    }
+
+    /**
+     * * 判断一个Collection是否为空, 包含List,Set,Queue
+     * 
+     * @param coll 要判断的Collection
+     * @return true:为空 false:非空
+     */
+    public static boolean isEmpty(Collection<?> coll)
+    {
+        return isNull(coll) || coll.isEmpty();
+    }
+
+    /**
+     * * 判断一个Collection是否非空,包含List,Set,Queue
+     * 
+     * @param coll 要判断的Collection
+     * @return true:非空 false:空
+     */
+    public static boolean isNotEmpty(Collection<?> coll)
+    {
+        return !isEmpty(coll);
+    }
+
+    /**
+     * * 判断一个对象数组是否为空
+     * 
+     * @param objects 要判断的对象数组
+     ** @return true:为空 false:非空
+     */
+    public static boolean isEmpty(Object[] objects)
+    {
+        return isNull(objects) || (objects.length == 0);
+    }
+
+    /**
+     * * 判断一个对象数组是否非空
+     * 
+     * @param objects 要判断的对象数组
+     * @return true:非空 false:空
+     */
+    public static boolean isNotEmpty(Object[] objects)
+    {
+        return !isEmpty(objects);
+    }
+
+    /**
+     * * 判断一个Map是否为空
+     * 
+     * @param map 要判断的Map
+     * @return true:为空 false:非空
+     */
+    public static boolean isEmpty(Map<?, ?> map)
+    {
+        return isNull(map) || map.isEmpty();
+    }
+
+    /**
+     * * 判断一个Map是否为空
+     * 
+     * @param map 要判断的Map
+     * @return true:非空 false:空
+     */
+    public static boolean isNotEmpty(Map<?, ?> map)
+    {
+        return !isEmpty(map);
+    }
+
+    /**
+     * * 判断一个字符串是否为空串
+     * 
+     * @param str String
+     * @return true:为空 false:非空
+     */
+    public static boolean isEmpty(String str)
+    {
+        return isNull(str) || NULLSTR.equals(str.trim());
+    }
+
+    /**
+     * * 判断一个字符串是否为非空串
+     * 
+     * @param str String
+     * @return true:非空串 false:空串
+     */
+    public static boolean isNotEmpty(String str)
+    {
+        return !isEmpty(str);
+    }
+
+    /**
+     * * 判断一个对象是否为空
+     * 
+     * @param object Object
+     * @return true:为空 false:非空
+     */
+    public static boolean isNull(Object object)
+    {
+        return object == null;
+    }
+
+    /**
+     * * 判断一个对象是否非空
+     * 
+     * @param object Object
+     * @return true:非空 false:空
+     */
+    public static boolean isNotNull(Object object)
+    {
+        return !isNull(object);
+    }
+
+    /**
+     * * 判断一个对象是否是数组类型(Java基本型别的数组)
+     * 
+     * @param object 对象
+     * @return true:是数组 false:不是数组
+     */
+    public static boolean isArray(Object object)
+    {
+        return isNotNull(object) && object.getClass().isArray();
+    }
+
+    /**
+     * 去空格
+     */
+    public static String trim(String str)
+    {
+        return (str == null ? "" : str.trim());
+    }
+
+    /**
+     * 截取字符串
+     * 
+     * @param str 字符串
+     * @param start 开始
+     * @return 结果
+     */
+    public static String substring(final String str, int start)
+    {
+        if (str == null)
+        {
+            return NULLSTR;
+        }
+
+        if (start < 0)
+        {
+            start = str.length() + start;
+        }
+
+        if (start < 0)
+        {
+            start = 0;
+        }
+        if (start > str.length())
+        {
+            return NULLSTR;
+        }
+
+        return str.substring(start);
+    }
+
+    /**
+     * 截取字符串
+     * 
+     * @param str 字符串
+     * @param start 开始
+     * @param end 结束
+     * @return 结果
+     */
+    public static String substring(final String str, int start, int end)
+    {
+        if (str == null)
+        {
+            return NULLSTR;
+        }
+
+        if (end < 0)
+        {
+            end = str.length() + end;
+        }
+        if (start < 0)
+        {
+            start = str.length() + start;
+        }
+
+        if (end > str.length())
+        {
+            end = str.length();
+        }
+
+        if (start > end)
+        {
+            return NULLSTR;
+        }
+
+        if (start < 0)
+        {
+            start = 0;
+        }
+        if (end < 0)
+        {
+            end = 0;
+        }
+
+        return str.substring(start, end);
+    }
+
+    /**
+     * 格式化文本, {} 表示占位符<br>
+     * 此方法只是简单将占位符 {} 按照顺序替换为参数<br>
+     * 如果想输出 {} 使用 \\转义 { 即可,如果想输出 {} 之前的 \ 使用双转义符 \\\\ 即可<br>
+     * 例:<br>
+     * 通常使用:format("this is {} for {}", "a", "b") -> this is a for b<br>
+     * 转义{}: format("this is \\{} for {}", "a", "b") -> this is \{} for a<br>
+     * 转义\: format("this is \\\\{} for {}", "a", "b") -> this is \a for b<br>
+     * 
+     * @param template 文本模板,被替换的部分用 {} 表示
+     * @param params 参数值
+     * @return 格式化后的文本
+     */
+//    public static String format(String template, Object... params)
+//    {
+//        if (isEmpty(params) || isEmpty(template))
+//        {
+//            return template;
+//        }
+//        return StrFormatter.format(template, params);
+//    }
+
+    /**
+     * 字符串转set
+     * 
+     * @param str 字符串
+     * @param sep 分隔符
+     * @return set集合
+     */
+    public static final Set<String> str2Set(String str, String sep)
+    {
+        return new HashSet<String>(str2List(str, sep, true, false));
+    }
+
+    /**
+     * 字符串转list
+     * 
+     * @param str 字符串
+     * @param sep 分隔符
+     * @param filterBlank 过滤纯空白
+     * @param trim 去掉首尾空白
+     * @return list集合
+     */
+    public static final List<String> str2List(String str, String sep, boolean filterBlank, boolean trim)
+    {
+        List<String> list = new ArrayList<String>();
+        if (StringUtils.isEmpty(str))
+        {
+            return list;
+        }
+
+        // 过滤空白字符串
+        if (filterBlank && StringUtils.isBlank(str))
+        {
+            return list;
+        }
+        String[] split = str.split(sep);
+        for (String string : split)
+        {
+            if (filterBlank && StringUtils.isBlank(string))
+            {
+                continue;
+            }
+            if (trim)
+            {
+                string = string.trim();
+            }
+            list.add(string);
+        }
+
+        return list;
+    }
+
+    /**
+     * 下划线转驼峰命名
+     */
+    public static String toUnderScoreCase(String str)
+    {
+        if (str == null)
+        {
+            return null;
+        }
+        StringBuilder sb = new StringBuilder();
+        // 前置字符是否大写
+        boolean preCharIsUpperCase = true;
+        // 当前字符是否大写
+        boolean curreCharIsUpperCase = true;
+        // 下一字符是否大写
+        boolean nexteCharIsUpperCase = true;
+        for (int i = 0; i < str.length(); i++)
+        {
+            char c = str.charAt(i);
+            if (i > 0)
+            {
+                preCharIsUpperCase = Character.isUpperCase(str.charAt(i - 1));
+            }
+            else
+            {
+                preCharIsUpperCase = false;
+            }
+
+            curreCharIsUpperCase = Character.isUpperCase(c);
+
+            if (i < (str.length() - 1))
+            {
+                nexteCharIsUpperCase = Character.isUpperCase(str.charAt(i + 1));
+            }
+
+            if (preCharIsUpperCase && curreCharIsUpperCase && !nexteCharIsUpperCase)
+            {
+                sb.append(SEPARATOR);
+            }
+            else if ((i != 0 && !preCharIsUpperCase) && curreCharIsUpperCase)
+            {
+                sb.append(SEPARATOR);
+            }
+            sb.append(Character.toLowerCase(c));
+        }
+
+        return sb.toString();
+    }
+
+    /**
+     * 是否包含字符串
+     * 
+     * @param str 验证字符串
+     * @param strs 字符串组
+     * @return 包含返回true
+     */
+    public static boolean inStringIgnoreCase(String str, String... strs)
+    {
+        if (str != null && strs != null)
+        {
+            for (String s : strs)
+            {
+                if (str.equalsIgnoreCase(trim(s)))
+                {
+                    return true;
+                }
+            }
+        }
+        return false;
+    }
+
+    /**
+     * 将下划线大写方式命名的字符串转换为驼峰式。如果转换前的下划线大写方式命名的字符串为空,则返回空字符串。 例如:HELLO_WORLD->HelloWorld
+     * 
+     * @param name 转换前的下划线大写方式命名的字符串
+     * @return 转换后的驼峰式命名的字符串
+     */
+    public static String convertToCamelCase(String name)
+    {
+        StringBuilder result = new StringBuilder();
+        // 快速检查
+        if (name == null || name.isEmpty())
+        {
+            // 没必要转换
+            return "";
+        }
+        else if (!name.contains("_"))
+        {
+            // 不含下划线,仅将首字母大写
+            return name.substring(0, 1).toUpperCase() + name.substring(1);
+        }
+        // 用下划线将原始字符串分割
+        String[] camels = name.split("_");
+        for (String camel : camels)
+        {
+            // 跳过原始字符串中开头、结尾的下换线或双重下划线
+            if (camel.isEmpty())
+            {
+                continue;
+            }
+            // 首字母大写
+            result.append(camel.substring(0, 1).toUpperCase());
+            result.append(camel.substring(1).toLowerCase());
+        }
+        return result.toString();
+    }
+
+    /**
+     * 驼峰式命名法 例如:user_name->userName
+     */
+    public static String toCamelCase(String s)
+    {
+        if (s == null)
+        {
+            return null;
+        }
+        s = s.toLowerCase();
+        StringBuilder sb = new StringBuilder(s.length());
+        boolean upperCase = false;
+        for (int i = 0; i < s.length(); i++)
+        {
+            char c = s.charAt(i);
+
+            if (c == SEPARATOR)
+            {
+                upperCase = true;
+            }
+            else if (upperCase)
+            {
+                sb.append(Character.toUpperCase(c));
+                upperCase = false;
+            }
+            else
+            {
+                sb.append(c);
+            }
+        }
+        return sb.toString();
+    }
+
+    @SuppressWarnings("unchecked")
+    public static <T> T cast(Object obj)
+    {
+        return (T) obj;
+    }
+}

+ 112 - 0
UtilsDemo-WX/src/main/java/com/example/util/TokenUtil.java

@@ -0,0 +1,112 @@
+package com.example.util;
+
+import com.example.base.Token;
+import com.example.constant.GlobleConstant;
+import io.jsonwebtoken.Claims;
+import io.jsonwebtoken.JwtBuilder;
+import io.jsonwebtoken.Jwts;
+import io.jsonwebtoken.SignatureAlgorithm;
+
+import javax.crypto.spec.SecretKeySpec;
+import javax.xml.bind.DatatypeConverter;
+import java.security.Key;
+import java.sql.Timestamp;
+import java.util.Date;
+import java.util.HashMap;
+import java.util.Map;
+
+/**
+ * Token 生成工具类
+ *
+ * @author yys
+ */
+public class TokenUtil {
+
+    /**
+     * 签名秘钥
+     */
+    public static final String SECRET = "networkSecurity";
+    /**
+     * 签发者
+     */
+    public static final String ISSUER = "www.hywan.com";
+
+    /**
+     * 生成token
+     *
+     * @param id      用户Id
+     * @param subject 用户类型
+     * @author yys
+     */
+    public static Token createJwtToken(String id, String subject) {
+        // 生成签发时间
+        long nowMillis = System.currentTimeMillis();
+        Date issueDate = new Date(nowMillis);
+        Date tokenExpireDate = new Date(nowMillis + GlobleConstant.TOKEN_EXPIRES_SECOND * 1000);
+        Date refreshtokenExpireDate = new Date(
+                nowMillis + GlobleConstant.REFRESH_TOKEN_EXPIRES_SECOND * 1000);
+        String accessToken = createJwtToken(id, ISSUER, subject, issueDate, tokenExpireDate,
+                GlobleConstant.TOKEN_AUD);
+        String refreshToken = createJwtToken(id, ISSUER, subject, issueDate, refreshtokenExpireDate,
+                GlobleConstant.REFRESH_TOKEN_AUD);
+        Token token = new Token();
+        token.setExpireTime(new Timestamp(tokenExpireDate.getTime()));
+        token.setAccessToken(accessToken);
+        token.setRefreshToken(refreshToken);
+        return token;
+    }
+
+    /**
+     * 生成token
+     *
+     * @author yys
+     */
+    public static String createJwtToken(String id, String issuer, String subject, Date issueDate,
+                                        Date expireDate, String aud) {
+
+        // 签名算法 ,将对token进行签名
+        SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;
+
+        // 通过秘钥签名JWT
+        byte[] apiKeySecretBytes = DatatypeConverter.parseBase64Binary(SECRET);
+        Key signingKey = new SecretKeySpec(apiKeySecretBytes, signatureAlgorithm.getJcaName());
+        Map<String, Object> claims = new HashMap<>();
+        claims.put("uid", id);
+        JwtBuilder builder = Jwts.builder().setClaims(claims).setId(id).setIssuedAt(issueDate)
+                .setSubject(subject).setIssuer(issuer).setAudience(aud)
+                .signWith(signatureAlgorithm, signingKey);
+        builder.setExpiration(expireDate);
+        return builder.compact();
+    }
+
+    /**
+     * 解析验证token
+     *
+     * @author yys
+     */
+    public static Claims parseJWT(String token) {
+        Claims claims = Jwts.parser().setSigningKey(DatatypeConverter.parseBase64Binary(SECRET))
+                .parseClaimsJws(token)
+                .getBody();
+        return claims;
+    }
+
+    public static String getUserId(String token) {
+        try {
+            token = token.substring(0, token.length() - 32);
+            Claims claims = Jwts.parser().setSigningKey(DatatypeConverter.parseBase64Binary(SECRET)).parseClaimsJws
+                    (token)
+                    .getBody();
+            return claims.get("uid").toString();
+        } catch (Exception e) {
+            System.err.println("token 不合法===============>:" + token);
+            return null;
+        }
+    }
+
+//	public static void main(String[] args) {
+//		Claims claims = TokenUtil.parseJWT
+// ("eyJhbGciOiJIUzI1NiJ9.eyJqdGkiOiIyNzk2YjIzNmRiOGE0MjA5YTgyMGEzMTBiZmU1N2ZhNyIsImlhdCI6MTUzODM2Mzc2OSwic3ViIjoiQ3VycmVudE1lbWJlciIsImlzcyI6Ind3dy5oeXdhbi5jb20iLCJhdWQiOiJ0b2tlbkF1ZCIsImV4cCI6MTUzODk2ODU2OX0.Bd0FFqDyRcrjcWgUclo57N8TmR_rgZJ7-n6Amia7_0w");
+//		System.out.printf(claims.getSubject());
+//	}
+}

+ 486 - 0
UtilsDemo-WX/src/main/java/com/example/util/UUID.java

@@ -0,0 +1,486 @@
+package com.example.util;
+
+
+import com.sun.xml.internal.ws.util.UtilException;
+
+import java.security.MessageDigest;
+import java.security.NoSuchAlgorithmException;
+import java.security.SecureRandom;
+import java.util.Random;
+import java.util.concurrent.ThreadLocalRandom;
+
+/**
+ * 提供通用唯一识别码(universally unique identifier)(UUID)实现
+ *
+ * @author ruoyi
+ */
+public final class UUID implements java.io.Serializable, Comparable<UUID>
+{
+    private static final long serialVersionUID = -1185015143654744140L;
+
+    /**
+     * SecureRandom 的单例
+     *
+     */
+    private static class Holder
+    {
+        static final SecureRandom numberGenerator = getSecureRandom();
+    }
+
+    /** 此UUID的最高64有效位 */
+    private final long mostSigBits;
+
+    /** 此UUID的最低64有效位 */
+    private final long leastSigBits;
+
+    /**
+     * 私有构造
+     * 
+     * @param data 数据
+     */
+    private UUID(byte[] data)
+    {
+        long msb = 0;
+        long lsb = 0;
+        assert data.length == 16 : "data must be 16 bytes in length";
+        for (int i = 0; i < 8; i++)
+        {
+            msb = (msb << 8) | (data[i] & 0xff);
+        }
+        for (int i = 8; i < 16; i++)
+        {
+            lsb = (lsb << 8) | (data[i] & 0xff);
+        }
+        this.mostSigBits = msb;
+        this.leastSigBits = lsb;
+    }
+
+    /**
+     * 使用指定的数据构造新的 UUID。
+     *
+     * @param mostSigBits 用于 {@code UUID} 的最高有效 64 位
+     * @param leastSigBits 用于 {@code UUID} 的最低有效 64 位
+     */
+    public UUID(long mostSigBits, long leastSigBits)
+    {
+        this.mostSigBits = mostSigBits;
+        this.leastSigBits = leastSigBits;
+    }
+
+    /**
+     * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的本地线程伪随机数生成器生成该 UUID。
+     * 
+     * @return 随机生成的 {@code UUID}
+     */
+    public static UUID fastUUID()
+    {
+        return randomUUID(false);
+    }
+
+    /**
+     * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。
+     * 
+     * @return 随机生成的 {@code UUID}
+     */
+    public static UUID randomUUID()
+    {
+        return randomUUID(true);
+    }
+
+    /**
+     * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。
+     * 
+     * @param isSecure 是否使用{@link SecureRandom}如果是可以获得更安全的随机码,否则可以得到更好的性能
+     * @return 随机生成的 {@code UUID}
+     */
+    public static UUID randomUUID(boolean isSecure)
+    {
+        final Random ng = isSecure ? Holder.numberGenerator : getRandom();
+
+        byte[] randomBytes = new byte[16];
+        ng.nextBytes(randomBytes);
+        randomBytes[6] &= 0x0f; /* clear version */
+        randomBytes[6] |= 0x40; /* set to version 4 */
+        randomBytes[8] &= 0x3f; /* clear variant */
+        randomBytes[8] |= 0x80; /* set to IETF variant */
+        return new UUID(randomBytes);
+    }
+
+    /**
+     * 根据指定的字节数组获取类型 3(基于名称的)UUID 的静态工厂。
+     *
+     * @param name 用于构造 UUID 的字节数组。
+     *
+     * @return 根据指定数组生成的 {@code UUID}
+     */
+    public static UUID nameUUIDFromBytes(byte[] name)
+    {
+        MessageDigest md;
+        try
+        {
+            md = MessageDigest.getInstance("MD5");
+        }
+        catch (NoSuchAlgorithmException nsae)
+        {
+            throw new InternalError("MD5 not supported");
+        }
+        byte[] md5Bytes = md.digest(name);
+        md5Bytes[6] &= 0x0f; /* clear version */
+        md5Bytes[6] |= 0x30; /* set to version 3 */
+        md5Bytes[8] &= 0x3f; /* clear variant */
+        md5Bytes[8] |= 0x80; /* set to IETF variant */
+        return new UUID(md5Bytes);
+    }
+
+    /**
+     * 根据 {@link #toString()} 方法中描述的字符串标准表示形式创建{@code UUID}。
+     *
+     * @param name 指定 {@code UUID} 字符串
+     * @return 具有指定值的 {@code UUID}
+     * @throws IllegalArgumentException 如果 name 与 {@link #toString} 中描述的字符串表示形式不符抛出此异常
+     *
+     */
+    public static UUID fromString(String name)
+    {
+        String[] components = name.split("-");
+        if (components.length != 5)
+        {
+            throw new IllegalArgumentException("Invalid UUID string: " + name);
+        }
+        for (int i = 0; i < 5; i++)
+        {
+            components[i] = "0x" + components[i];
+        }
+
+        long mostSigBits = Long.decode(components[0]).longValue();
+        mostSigBits <<= 16;
+        mostSigBits |= Long.decode(components[1]).longValue();
+        mostSigBits <<= 16;
+        mostSigBits |= Long.decode(components[2]).longValue();
+
+        long leastSigBits = Long.decode(components[3]).longValue();
+        leastSigBits <<= 48;
+        leastSigBits |= Long.decode(components[4]).longValue();
+
+        return new UUID(mostSigBits, leastSigBits);
+    }
+
+    /**
+     * 返回此 UUID 的 128 位值中的最低有效 64 位。
+     *
+     * @return 此 UUID 的 128 位值中的最低有效 64 位。
+     */
+    public long getLeastSignificantBits()
+    {
+        return leastSigBits;
+    }
+
+    /**
+     * 返回此 UUID 的 128 位值中的最高有效 64 位。
+     *
+     * @return 此 UUID 的 128 位值中最高有效 64 位。
+     */
+    public long getMostSignificantBits()
+    {
+        return mostSigBits;
+    }
+
+    /**
+     * 与此 {@code UUID} 相关联的版本号. 版本号描述此 {@code UUID} 是如何生成的。
+     * <p>
+     * 版本号具有以下含意:
+     * <ul>
+     * <li>1 基于时间的 UUID
+     * <li>2 DCE 安全 UUID
+     * <li>3 基于名称的 UUID
+     * <li>4 随机生成的 UUID
+     * </ul>
+     *
+     * @return 此 {@code UUID} 的版本号
+     */
+    public int version()
+    {
+        // Version is bits masked by 0x000000000000F000 in MS long
+        return (int) ((mostSigBits >> 12) & 0x0f);
+    }
+
+    /**
+     * 与此 {@code UUID} 相关联的变体号。变体号描述 {@code UUID} 的布局。
+     * <p>
+     * 变体号具有以下含意:
+     * <ul>
+     * <li>0 为 NCS 向后兼容保留
+     * <li>2 <a href="http://www.ietf.org/rfc/rfc4122.txt">IETF&nbsp;RFC&nbsp;4122</a>(Leach-Salz), 用于此类
+     * <li>6 保留,微软向后兼容
+     * <li>7 保留供以后定义使用
+     * </ul>
+     *
+     * @return 此 {@code UUID} 相关联的变体号
+     */
+    public int variant()
+    {
+        // This field is composed of a varying number of bits.
+        // 0 - - Reserved for NCS backward compatibility
+        // 1 0 - The IETF aka Leach-Salz variant (used by this class)
+        // 1 1 0 Reserved, Microsoft backward compatibility
+        // 1 1 1 Reserved for future definition.
+        return (int) ((leastSigBits >>> (64 - (leastSigBits >>> 62))) & (leastSigBits >> 63));
+    }
+
+    /**
+     * 与此 UUID 相关联的时间戳值。
+     *
+     * <p>
+     * 60 位的时间戳值根据此 {@code UUID} 的 time_low、time_mid 和 time_hi 字段构造。<br>
+     * 所得到的时间戳以 100 毫微秒为单位,从 UTC(通用协调时间) 1582 年 10 月 15 日零时开始。
+     *
+     * <p>
+     * 时间戳值仅在在基于时间的 UUID(其 version 类型为 1)中才有意义。<br>
+     * 如果此 {@code UUID} 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。
+     *
+     * @throws UnsupportedOperationException 如果此 {@code UUID} 不是 version 为 1 的 UUID。
+     */
+    public long timestamp() throws UnsupportedOperationException
+    {
+        checkTimeBase();
+        return (mostSigBits & 0x0FFFL) << 48//
+                | ((mostSigBits >> 16) & 0x0FFFFL) << 32//
+                | mostSigBits >>> 32;
+    }
+
+    /**
+     * 与此 UUID 相关联的时钟序列值。
+     *
+     * <p>
+     * 14 位的时钟序列值根据此 UUID 的 clock_seq 字段构造。clock_seq 字段用于保证在基于时间的 UUID 中的时间唯一性。
+     * <p>
+     * {@code clockSequence} 值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。 如果此 UUID 不是基于时间的 UUID,则此方法抛出
+     * UnsupportedOperationException。
+     *
+     * @return 此 {@code UUID} 的时钟序列
+     *
+     * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1
+     */
+    public int clockSequence() throws UnsupportedOperationException
+    {
+        checkTimeBase();
+        return (int) ((leastSigBits & 0x3FFF000000000000L) >>> 48);
+    }
+
+    /**
+     * 与此 UUID 相关的节点值。
+     *
+     * <p>
+     * 48 位的节点值根据此 UUID 的 node 字段构造。此字段旨在用于保存机器的 IEEE 802 地址,该地址用于生成此 UUID 以保证空间唯一性。
+     * <p>
+     * 节点值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。<br>
+     * 如果此 UUID 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。
+     *
+     * @return 此 {@code UUID} 的节点值
+     *
+     * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1
+     */
+    public long node() throws UnsupportedOperationException
+    {
+        checkTimeBase();
+        return leastSigBits & 0x0000FFFFFFFFFFFFL;
+    }
+
+    /**
+     * 返回此{@code UUID} 的字符串表现形式。
+     *
+     * <p>
+     * UUID 的字符串表示形式由此 BNF 描述:
+     * 
+     * <pre>
+     * {@code
+     * UUID                   = <time_low>-<time_mid>-<time_high_and_version>-<variant_and_sequence>-<node>
+     * time_low               = 4*<hexOctet>
+     * time_mid               = 2*<hexOctet>
+     * time_high_and_version  = 2*<hexOctet>
+     * variant_and_sequence   = 2*<hexOctet>
+     * node                   = 6*<hexOctet>
+     * hexOctet               = <hexDigit><hexDigit>
+     * hexDigit               = [0-9a-fA-F]
+     * }
+     * </pre>
+     * 
+     * </blockquote>
+     *
+     * @return 此{@code UUID} 的字符串表现形式
+     * @see #toString(boolean)
+     */
+    @Override
+    public String toString()
+    {
+        return toString(false);
+    }
+
+    /**
+     * 返回此{@code UUID} 的字符串表现形式。
+     *
+     * <p>
+     * UUID 的字符串表示形式由此 BNF 描述:
+     * 
+     * <pre>
+     * {@code
+     * UUID                   = <time_low>-<time_mid>-<time_high_and_version>-<variant_and_sequence>-<node>
+     * time_low               = 4*<hexOctet>
+     * time_mid               = 2*<hexOctet>
+     * time_high_and_version  = 2*<hexOctet>
+     * variant_and_sequence   = 2*<hexOctet>
+     * node                   = 6*<hexOctet>
+     * hexOctet               = <hexDigit><hexDigit>
+     * hexDigit               = [0-9a-fA-F]
+     * }
+     * </pre>
+     * 
+     * </blockquote>
+     *
+     * @param isSimple 是否简单模式,简单模式为不带'-'的UUID字符串
+     * @return 此{@code UUID} 的字符串表现形式
+     */
+    public String toString(boolean isSimple)
+    {
+        final StringBuilder builder = new StringBuilder(isSimple ? 32 : 36);
+        // time_low
+        builder.append(digits(mostSigBits >> 32, 8));
+        if (false == isSimple)
+        {
+            builder.append('-');
+        }
+        // time_mid
+        builder.append(digits(mostSigBits >> 16, 4));
+        if (false == isSimple)
+        {
+            builder.append('-');
+        }
+        // time_high_and_version
+        builder.append(digits(mostSigBits, 4));
+        if (false == isSimple)
+        {
+            builder.append('-');
+        }
+        // variant_and_sequence
+        builder.append(digits(leastSigBits >> 48, 4));
+        if (false == isSimple)
+        {
+            builder.append('-');
+        }
+        // node
+        builder.append(digits(leastSigBits, 12));
+
+        return builder.toString();
+    }
+
+    /**
+     * 返回此 UUID 的哈希码。
+     *
+     * @return UUID 的哈希码值。
+     */
+    @Override
+    public int hashCode()
+    {
+        long hilo = mostSigBits ^ leastSigBits;
+        return ((int) (hilo >> 32)) ^ (int) hilo;
+    }
+
+    /**
+     * 将此对象与指定对象比较。
+     * <p>
+     * 当且仅当参数不为 {@code null}、而是一个 UUID 对象、具有与此 UUID 相同的 varriant、包含相同的值(每一位均相同)时,结果才为 {@code true}。
+     *
+     * @param obj 要与之比较的对象
+     *
+     * @return 如果对象相同,则返回 {@code true};否则返回 {@code false}
+     */
+    @Override
+    public boolean equals(Object obj)
+    {
+        if ((null == obj) || (obj.getClass() != UUID.class))
+        {
+            return false;
+        }
+        UUID id = (UUID) obj;
+        return (mostSigBits == id.mostSigBits && leastSigBits == id.leastSigBits);
+    }
+
+    // Comparison Operations
+
+    /**
+     * 将此 UUID 与指定的 UUID 比较。
+     *
+     * <p>
+     * 如果两个 UUID 不同,且第一个 UUID 的最高有效字段大于第二个 UUID 的对应字段,则第一个 UUID 大于第二个 UUID。
+     *
+     * @param val 与此 UUID 比较的 UUID
+     *
+     * @return 在此 UUID 小于、等于或大于 val 时,分别返回 -1、0 或 1。
+     *
+     */
+    @Override
+    public int compareTo(UUID val)
+    {
+        // The ordering is intentionally set up so that the UUIDs
+        // can simply be numerically compared as two numbers
+        return (this.mostSigBits < val.mostSigBits ? -1 : //
+                (this.mostSigBits > val.mostSigBits ? 1 : //
+                        (this.leastSigBits < val.leastSigBits ? -1 : //
+                                (this.leastSigBits > val.leastSigBits ? 1 : //
+                                        0))));
+    }
+
+    // -------------------------------------------------------------------------------------------------------------------
+    // Private method start
+    /**
+     * 返回指定数字对应的hex值
+     * 
+     * @param val 值
+     * @param digits 位
+     * @return 值
+     */
+    private static String digits(long val, int digits)
+    {
+        long hi = 1L << (digits * 4);
+        return Long.toHexString(hi | (val & (hi - 1))).substring(1);
+    }
+
+    /**
+     * 检查是否为time-based版本UUID
+     */
+    private void checkTimeBase()
+    {
+        if (version() != 1)
+        {
+            throw new UnsupportedOperationException("Not a time-based UUID");
+        }
+    }
+
+    /**
+     * 获取{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
+     * 
+     * @return {@link SecureRandom}
+     */
+    public static SecureRandom getSecureRandom()
+    {
+        try
+        {
+            return SecureRandom.getInstance("SHA1PRNG");
+        }
+        catch (NoSuchAlgorithmException e)
+        {
+            throw new UtilException(e);
+        }
+    }
+
+    /**
+     * 获取随机数生成器对象<br>
+     * ThreadLocalRandom是JDK 7之后提供并发产生随机数,能够解决多个线程发生的竞争争夺。
+     * 
+     * @return {@link ThreadLocalRandom}
+     */
+    public static ThreadLocalRandom getRandom()
+    {
+        return ThreadLocalRandom.current();
+    }
+}

+ 69 - 0
UtilsDemo-WX/src/main/java/com/example/util/UUIDUtils.java

@@ -0,0 +1,69 @@
+package com.example.util;
+
+import java.text.SimpleDateFormat;
+import java.util.Date;
+import java.util.Random;
+import java.util.UUID;
+
+public class UUIDUtils {
+    public static String randomUUID() {
+        String uuid = UUID.randomUUID().toString();
+        return replace(uuid);
+    }
+
+    public static String replace(String uuid) {
+        return uuid.replaceAll("-", "");
+    }
+
+    /**
+     * 生成订单信息16位订单
+     * @return
+     */
+    public synchronized static String randomOrder(){
+        String code="";
+        Random rand=new Random();//生成随机数
+        for(int a=0;a<4;a++){
+            code+=rand.nextInt(10);//生成4位验证码
+        }
+        SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
+        String orderNo = sdf.format(new Date()).substring(2)+code;
+        return orderNo;
+    }
+
+    /**
+     * 获取随机码
+     * @return
+     */
+    public static String randomOrderLeng(Integer leng){
+        String code="";
+        Random rand=new Random();//生成随机数
+        for(int a=0;a<leng;a++){
+            code+=rand.nextInt(10);//生成6位验证码
+        }
+        SimpleDateFormat sdf = new SimpleDateFormat("SSS");
+        String orderNo = "G"+sdf.format(new Date())+code;
+        return orderNo;
+    }
+    /**
+     * 获取随机码
+     * @return
+     */
+    public static String randomCertLeng(Integer leng){
+        String code = "";
+        Random rand=new Random();//生成随机数
+        for(int a=0;a<leng;a++){
+            code+=rand.nextInt(10);//生成6位验证码
+        }
+        return code;
+    }
+    public static String setCodeString(Integer lengs,String leng) {
+        String s = "";
+        for(int i = 1 ;i < leng.length();i++){
+            s+="0";
+        }
+        Integer n = lengs - leng.length();
+        String code = UUIDUtils.randomCertLeng(n)+s;
+        return code;
+    }
+
+}

+ 55 - 0
UtilsDemo-WX/src/main/java/com/example/util/WxUtils.java

@@ -0,0 +1,55 @@
+package com.example.util;
+
+
+import org.apache.commons.codec.binary.Base64;
+
+import javax.crypto.Cipher;
+import javax.crypto.spec.IvParameterSpec;
+import javax.crypto.spec.SecretKeySpec;
+import java.security.spec.AlgorithmParameterSpec;
+
+/**
+ * by zdd
+ */
+public class WxUtils {
+
+    private static String APP_ID = "小程序APPID";
+    private static String SECRET = "小程序SECRET";
+
+    String openId = "";
+
+
+    public static String url = "https://api.weixin.qq.com/sns/jscode2session?appid=" + APP_ID + "&secret=" + SECRET + "&js_code=";
+    public static String url0 = "&grant_type=authorization_code";
+
+    // 步数转卡路里
+    public static float getDistanceByStep(long steps) {
+        return steps * 0.6f / 1000;
+    }
+
+    /**
+     * 微信解密运动步数
+     *
+     * @param sessionKey
+     * @param encryptedData
+     * @param iv
+     * @return
+     */
+    public static String decryptWeChatRunInfo(String sessionKey, String encryptedData, String iv) {
+        String result = null;
+        byte[] encrypData = Base64.decodeBase64(encryptedData);
+        byte[] ivData = Base64.decodeBase64(iv);
+        byte[] sessionKeyB = Base64.decodeBase64(sessionKey);
+        try {
+            AlgorithmParameterSpec ivSpec = new IvParameterSpec(ivData);
+            Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
+            SecretKeySpec keySpec = new SecretKeySpec(sessionKeyB, "AES");
+            cipher.init(Cipher.DECRYPT_MODE, keySpec, ivSpec);
+            byte[] doFinal = cipher.doFinal(encrypData);
+            result = new String(doFinal);
+        } catch (Exception e) {
+            e.printStackTrace();
+        }
+        return result;
+    }
+}

+ 22 - 0
UtilsDemo-WX/src/main/java/com/example/vo/UserUpdateVO.java

@@ -0,0 +1,22 @@
+package com.example.vo;
+
+import io.swagger.annotations.ApiModel;
+import io.swagger.annotations.ApiModelProperty;
+import lombok.Data;
+
+import javax.validation.constraints.NotBlank;
+import javax.validation.constraints.NotNull;
+
+@Data
+@ApiModel
+public class UserUpdateVO {
+
+    @ApiModelProperty(value = "手机号")
+    @NotNull(message = "手机号不能为空")
+    private String phone;
+
+    @ApiModelProperty(value = "验证码")
+    @NotNull(message = "验证码不能为空")
+    private String code;
+
+}

BIN
UtilsDemo-WX/src/main/resources/apiclient_cert.p12


+ 58 - 0
UtilsDemo-WX/src/main/resources/application.yml

@@ -0,0 +1,58 @@
+server:
+  port: 8899
+
+
+
+wx:
+  mp:
+   appId: wxbe90cc7c5233dd84
+   secret: ec4a18ce1a7bcf17cf592c3d57cec68a
+   url: http://wx.hw.hongweisoft.com/wxpaydemoapi/wechat/h5/userInfo
+   token: 123
+   aesKey: 123
+   before: http://wx.hw.hongweisoft.com/wxpaydemoapi/wechat/h5/authorize?returnUrl=http://wx.hw.hongweisoft.com/wxpaydemoh5
+  ma:
+   appId: wxbe90cc7c5233dd84
+   secret: ec4a18ce1a7bcf17cf592c3d57cec68a
+  pay:
+    wxpayAppcallbackurl: http://wx.hw.hongweisoft.com/wxpaydemoapi/app/payResult
+    wxpayRefundCallBackUrl: http://wx.hw.hongweisoft.com/wxpaydemoapi/app/refundResult
+    mchId: 1583849731
+    appId: wxbe90cc7c5233dd84
+    mchKey: B5F5A628C91D869F4CA357D2BCD63006
+    keyPath: classpath:apiclient_cert.p12
+
+spring:
+  redis:
+    host: 127.0.0.1
+    port: 6379
+    password:
+    database: 0
+    timeout: 10000
+  datasource:
+    driver-class-name: com.mysql.cj.jdbc.Driver
+    username: root
+    password: 123456
+    url: jdbc:mysql://localhost:3306/demo?characterEncoding=utf8&useSSL=false&serverTimezone=UTC&rewriteBatchedStatements=true
+    type: com.zaxxer.hikari.HikariDataSource
+    hikari:
+      pool-name: HikariCP
+      minimum-idle: 10
+      idle-timeout: 600000
+      maximum-pool-size: 50
+      auto-commit: true
+      max-lifetime: 1800000
+      connection-timeout: 30000
+
+jwt:
+  tokenHeader: Authorization
+  secret: forest-admin-secret
+  expiration: 604800
+  tokenHead: Bearer
+
+
+mybatis-plus:
+  configuration:
+    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
+  mapper-locations: classpath*:mapper/*.xml
+  type-aliases-package: com.example.entity

+ 4 - 0
UtilsDemo-WX/src/main/resources/mapper/User.xml

@@ -0,0 +1,4 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
+<mapper namespace="com.example.mapper.UserMapper">
+</mapper>

+ 44 - 0
UtilsDemo.iml

@@ -0,0 +1,44 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<module org.jetbrains.idea.maven.project.MavenProjectsManager.isMavenModule="true" type="JAVA_MODULE" version="4">
+  <component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_8">
+    <output url="file://$MODULE_DIR$/target/classes" />
+    <output-test url="file://$MODULE_DIR$/target/test-classes" />
+    <content url="file://$MODULE_DIR$">
+      <excludeFolder url="file://$MODULE_DIR$/target" />
+    </content>
+    <orderEntry type="inheritedJdk" />
+    <orderEntry type="sourceFolder" forTests="false" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-web:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-autoconfigure:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-logging:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-classic:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: ch.qos.logback:logback-core:1.2.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:slf4j-api:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-to-slf4j:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.logging.log4j:log4j-api:2.13.3" level="project" />
+    <orderEntry type="library" name="Maven: org.slf4j:jul-to-slf4j:1.7.30" level="project" />
+    <orderEntry type="library" name="Maven: jakarta.annotation:jakarta.annotation-api:1.3.5" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-core:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-jcl:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.yaml:snakeyaml:1.26" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-json:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-databind:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-annotations:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.core:jackson-core:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jdk8:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.datatype:jackson-datatype-jsr310:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: com.fasterxml.jackson.module:jackson-module-parameter-names:2.11.4" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework.boot:spring-boot-starter-tomcat:2.3.9.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-core:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.glassfish:jakarta.el:3.0.3" level="project" />
+    <orderEntry type="library" name="Maven: org.apache.tomcat.embed:tomcat-embed-websocket:9.0.43" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-web:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-beans:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-webmvc:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-aop:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-context:5.2.13.RELEASE" level="project" />
+    <orderEntry type="library" name="Maven: org.springframework:spring-expression:5.2.13.RELEASE" level="project" />
+  </component>
+</module>

+ 11 - 10
pom.xml

@@ -20,6 +20,7 @@
   <modules>
     <module>UtilsDemo-Excel</module>
     <module>UtilsDemo-SecureEncryption</module>
+    <module>UtilsDemo-WX</module>
   </modules>
   <packaging>pom</packaging>
 
@@ -38,16 +39,16 @@
 
 
   <build>
-    <resources>
-      <resource>
-        <directory>lib</directory>
-        <targetPath>/BOOT-INF/lib/</targetPath>
-        <includes>
-          <include>**/*.jar</include>
-
-        </includes>
-      </resource>
-    </resources>
+<!--    <resources>-->
+<!--      <resource>-->
+<!--        <directory>lib</directory>-->
+<!--        <targetPath>/BOOT-INF/lib/</targetPath>-->
+<!--        <includes>-->
+<!--          <include>**/*.jar</include>-->
+
+<!--        </includes>-->
+<!--      </resource>-->
+<!--    </resources>-->
 
     <pluginManagement>
       <plugins>

+ 168 - 0
target/site/css/maven-base.css

@@ -0,0 +1,168 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License.  You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied.  See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+body {
+  margin: 0px;
+  padding: 0px;
+}
+table {
+  padding:0px;
+  width: 100%;
+  margin-left: -2px;
+  margin-right: -2px;
+}
+acronym {
+  cursor: help;
+  border-bottom: 1px dotted #feb;
+}
+table.bodyTable th, table.bodyTable td {
+  padding: 2px 4px 2px 4px;
+  vertical-align: top;
+}
+div.clear{
+  clear:both;
+  visibility: hidden;
+}
+div.clear hr{
+  display: none;
+}
+#bannerLeft, #bannerRight {
+  font-size: xx-large;
+  font-weight: bold;
+}
+#bannerLeft img, #bannerRight img {
+  margin: 0px;
+}
+.xleft, #bannerLeft img {
+  float:left;
+}
+.xright, #bannerRight {
+  float:right;
+}
+#banner {
+  padding: 0px;
+}
+#breadcrumbs {
+  padding: 3px 10px 3px 10px;
+}
+#leftColumn {
+ width: 170px;
+ float:left;
+ overflow: auto;
+}
+#bodyColumn {
+  margin-right: 1.5em;
+  margin-left: 197px;
+}
+#legend {
+  padding: 8px 0 8px 0;
+}
+#navcolumn {
+  padding: 8px 4px 0 8px;
+}
+#navcolumn h5 {
+  margin: 0;
+  padding: 0;
+  font-size: small;
+}
+#navcolumn ul {
+  margin: 0;
+  padding: 0;
+  font-size: small;
+}
+#navcolumn li {
+  list-style-type: none;
+  background-image: none;
+  background-repeat: no-repeat;
+  background-position: 0 0.4em;
+  padding-left: 16px;
+  list-style-position: outside;
+  line-height: 1.2em;
+  font-size: smaller;
+}
+#navcolumn li.expanded {
+  background-image: url(../images/expanded.gif);
+}
+#navcolumn li.collapsed {
+  background-image: url(../images/collapsed.gif);
+}
+#navcolumn li.none {
+  text-indent: -1em;
+  margin-left: 1em;
+}
+#poweredBy {
+  text-align: center;
+}
+#navcolumn img {
+  margin-top: 10px;
+  margin-bottom: 3px;
+}
+#poweredBy img {
+  display:block;
+  margin: 20px 0 20px 17px;
+}
+#search img {
+    margin: 0px;
+    display: block;
+}
+#search #q, #search #btnG {
+    border: 1px solid #999;
+    margin-bottom:10px;
+}
+#search form {
+    margin: 0px;
+}
+#lastPublished {
+  font-size: x-small;
+}
+.navSection {
+  margin-bottom: 2px;
+  padding: 8px;
+}
+.navSectionHead {
+  font-weight: bold;
+  font-size: x-small;
+}
+.section {
+  padding: 4px;
+}
+#footer {
+  padding: 3px 10px 3px 10px;
+  font-size: x-small;
+}
+#breadcrumbs {
+  font-size: x-small;
+  margin: 0pt;
+}
+.source {
+  padding: 12px;
+  margin: 1em 7px 1em 7px;
+}
+.source pre {
+  margin: 0px;
+  padding: 0px;
+}
+#navcolumn img.imageLink, .imageLink {
+  padding-left: 0px;
+  padding-bottom: 0px;
+  padding-top: 0px;
+  padding-right: 2px;
+  border: 0px;
+  margin: 0px;
+}

+ 161 - 0
target/site/css/maven-theme.css

@@ -0,0 +1,161 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License.  You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied.  See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+body {
+  padding: 0px 0px 10px 0px;
+}
+body, td, select, input, li{
+  font-family: Verdana, Helvetica, Arial, sans-serif;
+  font-size: 13px;
+}
+code{
+  font-family: Courier, monospace;
+  font-size: 13px;
+}
+a {
+  text-decoration: none;
+}
+a:link {
+  color:#36a;
+}
+a:visited  {
+  color:#47a;
+}
+a:active, a:hover {
+  color:#69c;
+}
+#legend li.externalLink {
+  background: url(../images/external.png) left top no-repeat;
+  padding-left: 18px;
+}
+a.externalLink, a.externalLink:link, a.externalLink:visited, a.externalLink:active, a.externalLink:hover {
+  background: url(../images/external.png) right center no-repeat;
+  padding-right: 18px;
+}
+#legend li.newWindow {
+  background: url(../images/newwindow.png) left top no-repeat;
+  padding-left: 18px;
+}
+a.newWindow, a.newWindow:link, a.newWindow:visited, a.newWindow:active, a.newWindow:hover {
+  background: url(../images/newwindow.png) right center no-repeat;
+  padding-right: 18px;
+}
+h2 {
+  padding: 4px 4px 4px 6px;
+  border: 1px solid #999;
+  color: #900;
+  background-color: #ddd;
+  font-weight:900;
+  font-size: x-large;
+}
+h3 {
+  padding: 4px 4px 4px 6px;
+  border: 1px solid #aaa;
+  color: #900;
+  background-color: #eee;
+  font-weight: normal;
+  font-size: large;
+}
+h4 {
+  padding: 4px 4px 4px 6px;
+  border: 1px solid #bbb;
+  color: #900;
+  background-color: #fff;
+  font-weight: normal;
+  font-size: large;
+}
+h5 {
+  padding: 4px 4px 4px 6px;
+  color: #900;
+  font-size: medium;
+}
+p {
+  line-height: 1.3em;
+  font-size: small;
+}
+#breadcrumbs {
+  border-top: 1px solid #aaa;
+  border-bottom: 1px solid #aaa;
+  background-color: #ccc;
+}
+#leftColumn {
+  margin: 10px 0 0 5px;
+  border: 1px solid #999;
+  background-color: #eee;
+  padding-bottom: 3px; /* IE-9 scrollbar-fix */
+}
+#navcolumn h5 {
+  font-size: smaller;
+  border-bottom: 1px solid #aaaaaa;
+  padding-top: 2px;
+  color: #000;
+}
+
+table.bodyTable th {
+  color: white;
+  background-color: #bbb;
+  text-align: left;
+  font-weight: bold;
+}
+
+table.bodyTable th, table.bodyTable td {
+  font-size: 1em;
+}
+
+table.bodyTable tr.a {
+  background-color: #ddd;
+}
+
+table.bodyTable tr.b {
+  background-color: #eee;
+}
+
+.source {
+  border: 1px solid #999;
+}
+dl {
+  padding: 4px 4px 4px 6px;
+  border: 1px solid #aaa;
+  background-color: #ffc;
+}
+dt {
+  color: #900;
+}
+#organizationLogo img, #projectLogo img, #projectLogo span{
+  margin: 8px;
+}
+#banner {
+  border-bottom: 1px solid #fff;
+}
+.errormark, .warningmark, .donemark, .infomark {
+  background: url(../images/icon_error_sml.gif) no-repeat;
+}
+
+.warningmark {
+  background-image: url(../images/icon_warning_sml.gif);
+}
+
+.donemark {
+  background-image: url(../images/icon_success_sml.gif);
+}
+
+.infomark {
+  background-image: url(../images/icon_info_sml.gif);
+}
+

+ 26 - 0
target/site/css/print.css

@@ -0,0 +1,26 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements.  See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership.  The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License.  You may obtain a copy of the License at
+ *
+ *   http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied.  See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+#banner, #footer, #leftcol, #breadcrumbs, .docs #toc, .docs .courtesylinks, #leftColumn, #navColumn {
+	display: none !important;
+}
+#bodyColumn, body.docs div.docs {
+	margin: 0 !important;
+	border: none !important
+}

+ 1 - 0
target/site/css/site.css

@@ -0,0 +1 @@
+/* You can override this file with your own styles */

+ 903 - 0
target/site/dependencies.html

@@ -0,0 +1,903 @@
+<!DOCTYPE html>
+<!--
+ | Generated by Apache Maven Doxia Site Renderer 1.9.2 at 2021-03-11
+
+ | Rendered using Apache Maven Default Skin
+-->
+<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="zh" lang="zh">
+  <head>
+    <meta charset="UTF-8" />
+    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
+    <meta name="generator" content="Apache Maven Doxia Site Renderer 1.9.2" />
+    <title>项目依赖 &#x2013; 项目依赖</title>
+    <link rel="stylesheet" href="./css/maven-base.css" />
+    <link rel="stylesheet" href="./css/maven-theme.css" />
+    <link rel="stylesheet" href="./css/site.css" />
+    <link rel="stylesheet" href="./css/print.css" media="print" />
+  </head>
+  <body class="composite">
+    <div id="banner">
+      <div class="clear">
+        <hr/>
+      </div>
+    </div>
+    <div id="breadcrumbs">
+      <div class="xleft">
+        <span id="publishDate">最近更新: 2021-03-11</span>
+          &#xA0;| <span id="projectVersion">版本: 1.0-SNAPSHOT</span>
+      </div>
+      <div class="xright">      </div>
+      <div class="clear">
+        <hr/>
+      </div>
+    </div>
+    <div id="leftColumn">
+      <div id="navcolumn">
+      <a href="http://maven.apache.org/" title="构建依靠 Maven" class="poweredBy">
+        <img class="poweredBy" alt="构建依靠 Maven" src="./images/logos/maven-feather.png" />
+      </a>
+      </div>
+    </div>
+    <div id="bodyColumn">
+      <div id="contentBox">
+<a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96"></a><section>
+<h2><a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96"></a>&#x9879;&#x76ee;&#x4f9d;&#x8d56;</h2><a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96_compile"></a><section>
+<h3><a name="compile"></a>compile</h3>
+<p>&#x4ee5;&#x4e0b;&#x662f;&#x8fd9;&#x4e00;&#x9879;&#x76ee;&#x7684;&#x7f16;&#x8bd1;&#x4f9d;&#x8d56;&#x5217;&#x8868;&#x3002;&#x7f16;&#x8bd1;&#x548c;&#x8fd0;&#x884c;&#x672c;&#x5e94;&#x7528;&#x9700;&#x8981;&#x8fd9;&#x4e9b;&#x4f9d;&#x8d56;&#x3002;</p>
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>GroupId</th>
+<th>ArtifactId</th>
+<th>&#x7248;&#x672c;</th>
+<th>&#x7c7b;&#x578b;</th>
+<th>Licenses</th></tr>
+<tr class="b">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-starter-web</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr></table></section></section><a name="a.E9.A1.B9.E7.9B.AE.E4.BC.A0.E9.80.92.E4.BE.9D.E8.B5.96"></a><section>
+<h2><a name="a.E9.A1.B9.E7.9B.AE.E4.BC.A0.E9.80.92.E4.BE.9D.E8.B5.96"></a>&#x9879;&#x76ee;&#x4f20;&#x9012;&#x4f9d;&#x8d56;</h2>
+<p>&#x4ee5;&#x4e0b;&#x662f;&#x672c;&#x9879;&#x76ee;&#x7684;&#x4f20;&#x9012;&#x4f9d;&#x8d56;&#x5217;&#x8868;&#x3002;&#x4f20;&#x9012;&#x4f9d;&#x8d56;&#x662f;&#x9879;&#x76ee;&#x4f9d;&#x8d56;&#x7684;&#x4f9d;&#x8d56;&#xff1a;</p><a name="a.E9.A1.B9.E7.9B.AE.E4.BC.A0.E9.80.92.E4.BE.9D.E8.B5.96_compile"></a><section>
+<h3><a name="compile"></a>compile</h3>
+<p>&#x4ee5;&#x4e0b;&#x662f;&#x8fd9;&#x4e00;&#x9879;&#x76ee;&#x7684;&#x7f16;&#x8bd1;&#x4f9d;&#x8d56;&#x5217;&#x8868;&#x3002;&#x7f16;&#x8bd1;&#x548c;&#x8fd0;&#x884c;&#x672c;&#x5e94;&#x7528;&#x9700;&#x8981;&#x8fd9;&#x4e9b;&#x4f9d;&#x8d56;&#x3002;</p>
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>GroupId</th>
+<th>ArtifactId</th>
+<th>&#x7248;&#x672c;</th>
+<th>&#x7c7b;&#x578b;</th>
+<th>Licenses</th></tr>
+<tr class="b">
+<td>ch.qos.logback</td>
+<td><a class="externalLink" href="http://logback.qos.ch/logback-classic">logback-classic</a></td>
+<td>1.2.3</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.eclipse.org/legal/epl-v10.html">Eclipse Public License - v 1.0</a><a class="externalLink" href="http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html">GNU Lesser General Public License</a></td></tr>
+<tr class="a">
+<td>ch.qos.logback</td>
+<td><a class="externalLink" href="http://logback.qos.ch/logback-core">logback-core</a></td>
+<td>1.2.3</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.eclipse.org/legal/epl-v10.html">Eclipse Public License - v 1.0</a><a class="externalLink" href="http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html">GNU Lesser General Public License</a></td></tr>
+<tr class="b">
+<td>com.fasterxml.jackson.core</td>
+<td><a class="externalLink" href="http://github.com/FasterXML/jackson">jackson-annotations</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>com.fasterxml.jackson.core</td>
+<td><a class="externalLink" href="https://github.com/FasterXML/jackson-core">jackson-core</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>com.fasterxml.jackson.core</td>
+<td><a class="externalLink" href="http://github.com/FasterXML/jackson">jackson-databind</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>com.fasterxml.jackson.datatype</td>
+<td><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jdk8">jackson-datatype-jdk8</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>com.fasterxml.jackson.datatype</td>
+<td><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jsr310">jackson-datatype-jsr310</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>com.fasterxml.jackson.module</td>
+<td><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-module-parameter-names">jackson-module-parameter-names</a></td>
+<td>2.11.4</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>jakarta.annotation</td>
+<td><a class="externalLink" href="https://projects.eclipse.org/projects/ee4j.ca">jakarta.annotation-api</a></td>
+<td>1.3.5</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.eclipse.org/legal/epl-2.0">EPL 2.0</a><a class="externalLink" href="https://www.gnu.org/software/classpath/license.html">GPL2 w/ CPE</a></td></tr>
+<tr class="a">
+<td>org.apache.logging.log4j</td>
+<td><a class="externalLink" href="https://logging.apache.org/log4j/2.x/log4j-api/">log4j-api</a></td>
+<td>2.13.3</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.apache.logging.log4j</td>
+<td><a class="externalLink" href="https://logging.apache.org/log4j/2.x/log4j-to-slf4j/">log4j-to-slf4j</a></td>
+<td>2.13.3</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.apache.tomcat.embed</td>
+<td><a class="externalLink" href="https://tomcat.apache.org/">tomcat-embed-core</a></td>
+<td>9.0.43</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.apache.tomcat.embed</td>
+<td><a class="externalLink" href="https://tomcat.apache.org/">tomcat-embed-websocket</a></td>
+<td>9.0.43</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.glassfish</td>
+<td><a class="externalLink" href="https://projects.eclipse.org/projects/ee4j.el">jakarta.el</a></td>
+<td>3.0.3</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.eclipse.org/legal/epl-2.0">EPL 2.0</a><a class="externalLink" href="https://www.gnu.org/software/classpath/license.html">GPL2 w/ CPE</a></td></tr>
+<tr class="b">
+<td>org.slf4j</td>
+<td><a class="externalLink" href="http://www.slf4j.org">jul-to-slf4j</a></td>
+<td>1.7.30</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.opensource.org/licenses/mit-license.php">MIT License</a></td></tr>
+<tr class="a">
+<td>org.slf4j</td>
+<td><a class="externalLink" href="http://www.slf4j.org">slf4j-api</a></td>
+<td>1.7.30</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.opensource.org/licenses/mit-license.php">MIT License</a></td></tr>
+<tr class="b">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-aop</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-beans</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-context</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-core</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-expression</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-jcl</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-web</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework</td>
+<td><a class="externalLink" href="https://github.com/spring-projects/spring-framework">spring-webmvc</a></td>
+<td>5.2.13.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-autoconfigure</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-starter</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-starter-json</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-starter-logging</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="a">
+<td>org.springframework.boot</td>
+<td><a class="externalLink" href="https://spring.io/projects/spring-boot">spring-boot-starter-tomcat</a></td>
+<td>2.3.9.RELEASE</td>
+<td>jar</td>
+<td><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></td></tr>
+<tr class="b">
+<td>org.yaml</td>
+<td><a class="externalLink" href="http://www.snakeyaml.org">snakeyaml</a></td>
+<td>1.26</td>
+<td>jar</td>
+<td><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></td></tr></table></section></section><a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96.E5.9B.BE"></a><section>
+<h2><a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96.E5.9B.BE"></a>&#x9879;&#x76ee;&#x4f9d;&#x8d56;&#x56fe;</h2>
+<script language="javascript" type="text/javascript">
+      function toggleDependencyDetails( divId, imgId )
+      {
+        var div = document.getElementById( divId );
+        var img = document.getElementById( imgId );
+        if( div.style.display == '' )
+        {
+          div.style.display = 'none';
+          img.src='./images/icon_info_sml.gif';
+          img.alt='[Information]';
+        }
+        else
+        {
+          div.style.display = '';
+          img.src='./images/close.gif';
+          img.alt='[Close]';
+        }
+      }
+</script>
+<a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96.E6.A0.91"></a><section>
+<h3><a name="a.E9.A1.B9.E7.9B.AE.E4.BE.9D.E8.B5.96.E6.A0.91"></a>&#x9879;&#x76ee;&#x4f9d;&#x8d56;&#x6811;</h3>
+<ul>
+<li>org.example:UtilsDemo:pom:1.0-SNAPSHOT <img id="_img1" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep0', '_img1' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep0" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>UtilsDemo</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Parent pom providing dependency and plugin management for applications built with Maven</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot/UtilsDemo">https://spring.io/projects/spring-boot/UtilsDemo</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework.boot:spring-boot-starter-web:jar:2.3.9.RELEASE (compile) <img id="_img3" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep2', '_img3' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep2" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-starter-web</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Starter for building web, including RESTful, applications using Spring MVC. Uses Tomcat as the default embedded container</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework.boot:spring-boot-starter:jar:2.3.9.RELEASE (compile) <img id="_img5" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep4', '_img5' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep4" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-starter</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Core starter, including auto-configuration support, logging and YAML</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework.boot:spring-boot:jar:2.3.9.RELEASE (compile) <img id="_img7" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep6', '_img7' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep6" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Boot</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li>
+<li>org.springframework.boot:spring-boot-autoconfigure:jar:2.3.9.RELEASE (compile) <img id="_img9" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep8', '_img9' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep8" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-autoconfigure</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Boot AutoConfigure</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li>
+<li>org.springframework.boot:spring-boot-starter-logging:jar:2.3.9.RELEASE (compile) <img id="_img11" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep10', '_img11' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep10" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-starter-logging</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Starter for logging using Logback. Default logging starter</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>ch.qos.logback:logback-classic:jar:1.2.3 (compile) <img id="_img13" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep12', '_img13' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep12" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Logback Classic Module</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>logback-classic module</p>
+<p><b>URL: </b><a class="externalLink" href="http://logback.qos.ch/logback-classic">http://logback.qos.ch/logback-classic</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.eclipse.org/legal/epl-v10.html">Eclipse Public License - v 1.0</a>, <a class="externalLink" href="http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html">GNU Lesser General Public License</a></p></td></tr></table></div>
+<ul>
+<li>ch.qos.logback:logback-core:jar:1.2.3 (compile) <img id="_img15" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep14', '_img15' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep14" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Logback Core Module</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>logback-core module</p>
+<p><b>URL: </b><a class="externalLink" href="http://logback.qos.ch/logback-core">http://logback.qos.ch/logback-core</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.eclipse.org/legal/epl-v10.html">Eclipse Public License - v 1.0</a>, <a class="externalLink" href="http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html">GNU Lesser General Public License</a></p></td></tr></table></div></li>
+<li>org.slf4j:slf4j-api:jar:1.7.30 (compile) <img id="_img17" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep16', '_img17' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep16" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>SLF4J API Module</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>The slf4j API</p>
+<p><b>URL: </b><a class="externalLink" href="http://www.slf4j.org">http://www.slf4j.org</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.opensource.org/licenses/mit-license.php">MIT License</a></p></td></tr></table></div></li></ul></li>
+<li>org.apache.logging.log4j:log4j-to-slf4j:jar:2.13.3 (compile) <img id="_img19" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep18', '_img19' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep18" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Apache Log4j to SLF4J Adapter</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>The Apache Log4j binding between Log4j 2 API and SLF4J.</p>
+<p><b>URL: </b><a class="externalLink" href="https://logging.apache.org/log4j/2.x/log4j-to-slf4j/">https://logging.apache.org/log4j/2.x/log4j-to-slf4j/</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.apache.logging.log4j:log4j-api:jar:2.13.3 (compile) <img id="_img21" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep20', '_img21' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep20" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Apache Log4j API</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>The Apache Log4j API</p>
+<p><b>URL: </b><a class="externalLink" href="https://logging.apache.org/log4j/2.x/log4j-api/">https://logging.apache.org/log4j/2.x/log4j-api/</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.slf4j:jul-to-slf4j:jar:1.7.30 (compile) <img id="_img23" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep22', '_img23' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep22" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>JUL to SLF4J bridge</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>JUL to SLF4J bridge</p>
+<p><b>URL: </b><a class="externalLink" href="http://www.slf4j.org">http://www.slf4j.org</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.opensource.org/licenses/mit-license.php">MIT License</a></p></td></tr></table></div></li></ul></li>
+<li>jakarta.annotation:jakarta.annotation-api:jar:1.3.5 (compile) <img id="_img25" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep24', '_img25' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep24" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jakarta Annotations API</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Jakarta Annotations API</p>
+<p><b>URL: </b><a class="externalLink" href="https://projects.eclipse.org/projects/ee4j.ca">https://projects.eclipse.org/projects/ee4j.ca</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.eclipse.org/legal/epl-2.0">EPL 2.0</a>, <a class="externalLink" href="https://www.gnu.org/software/classpath/license.html">GPL2 w/ CPE</a></p></td></tr></table></div></li>
+<li>org.springframework:spring-core:jar:5.2.13.RELEASE (compile) <img id="_img27" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep26', '_img27' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep26" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Core</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Core</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework:spring-jcl:jar:5.2.13.RELEASE (compile) <img id="_img29" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep28', '_img29' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep28" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Commons Logging Bridge</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Commons Logging Bridge</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.yaml:snakeyaml:jar:1.26 (compile) <img id="_img31" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep30', '_img31' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep30" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>SnakeYAML</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>YAML 1.1 parser and emitter for Java</p>
+<p><b>URL: </b><a class="externalLink" href="http://www.snakeyaml.org">http://www.snakeyaml.org</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.springframework.boot:spring-boot-starter-json:jar:2.3.9.RELEASE (compile) <img id="_img33" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep32', '_img33' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep32" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-starter-json</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Starter for reading and writing json</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>com.fasterxml.jackson.core:jackson-databind:jar:2.11.4 (compile) <img id="_img35" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep34', '_img35' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep34" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>jackson-databind</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>General data-binding functionality for Jackson: works on core streaming API</p>
+<p><b>URL: </b><a class="externalLink" href="http://github.com/FasterXML/jackson">http://github.com/FasterXML/jackson</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>com.fasterxml.jackson.core:jackson-annotations:jar:2.11.4 (compile) <img id="_img37" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep36', '_img37' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep36" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jackson-annotations</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Core annotations used for value types, used by Jackson data binding package.</p>
+<p><b>URL: </b><a class="externalLink" href="http://github.com/FasterXML/jackson">http://github.com/FasterXML/jackson</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div></li>
+<li>com.fasterxml.jackson.core:jackson-core:jar:2.11.4 (compile) <img id="_img39" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep38', '_img39' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep38" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jackson-core</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Core Jackson processing abstractions (aka Streaming API), implementation for JSON</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/FasterXML/jackson-core">https://github.com/FasterXML/jackson-core</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>com.fasterxml.jackson.datatype:jackson-datatype-jdk8:jar:2.11.4 (compile) <img id="_img41" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep40', '_img41' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep40" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jackson datatype: jdk8</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Add-on module for Jackson (http://jackson.codehaus.org) to support
+JDK 8 data types.</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jdk8">https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jdk8</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div></li>
+<li>com.fasterxml.jackson.datatype:jackson-datatype-jsr310:jar:2.11.4 (compile) <img id="_img43" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep42', '_img43' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep42" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jackson datatype: JSR310</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Add-on module to support JSR-310 (Java 8 Date &amp; Time API) data types.</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jsr310">https://github.com/FasterXML/jackson-modules-java8/jackson-datatype-jsr310</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div></li>
+<li>com.fasterxml.jackson.module:jackson-module-parameter-names:jar:2.11.4 (compile) <img id="_img45" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep44', '_img45' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep44" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jackson-module-parameter-names</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Add-on module for Jackson (http://jackson.codehaus.org) to support
+introspection of method/constructor parameter names, without having to add explicit property name annotation.</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/FasterXML/jackson-modules-java8/jackson-module-parameter-names">https://github.com/FasterXML/jackson-modules-java8/jackson-module-parameter-names</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">The Apache Software License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.springframework.boot:spring-boot-starter-tomcat:jar:2.3.9.RELEASE (compile) <img id="_img47" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep46', '_img47' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep46" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>spring-boot-starter-tomcat</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Starter for using Tomcat as the embedded servlet container. Default servlet container starter used by spring-boot-starter-web</p>
+<p><b>URL: </b><a class="externalLink" href="https://spring.io/projects/spring-boot">https://spring.io/projects/spring-boot</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.apache.tomcat.embed:tomcat-embed-core:jar:9.0.43 (compile) <img id="_img49" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep48', '_img49' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep48" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>tomcat-embed-core</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Core Tomcat implementation</p>
+<p><b>URL: </b><a class="externalLink" href="https://tomcat.apache.org/">https://tomcat.apache.org/</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></p></td></tr></table></div></li>
+<li>org.glassfish:jakarta.el:jar:3.0.3 (compile) <img id="_img51" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep50', '_img51' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep50" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Jakarta Expression Language 3.0</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Jakarta Expression Language provides a specification document, API, reference implementation and TCK 
+        that describes an expression language for Java applications.</p>
+<p><b>URL: </b><a class="externalLink" href="https://projects.eclipse.org/projects/ee4j.el">https://projects.eclipse.org/projects/ee4j.el</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.eclipse.org/legal/epl-2.0">EPL 2.0</a>, <a class="externalLink" href="https://www.gnu.org/software/classpath/license.html">GPL2 w/ CPE</a></p></td></tr></table></div></li>
+<li>org.apache.tomcat.embed:tomcat-embed-websocket:jar:9.0.43 (compile) <img id="_img53" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep52', '_img53' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep52" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>tomcat-embed-websocket</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Core Tomcat implementation</p>
+<p><b>URL: </b><a class="externalLink" href="https://tomcat.apache.org/">https://tomcat.apache.org/</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="http://www.apache.org/licenses/LICENSE-2.0.txt">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.springframework:spring-web:jar:5.2.13.RELEASE (compile) <img id="_img55" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep54', '_img55' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep54" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Web</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Web</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework:spring-beans:jar:5.2.13.RELEASE (compile) <img id="_img57" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep56', '_img57' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep56" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Beans</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Beans</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li>
+<li>org.springframework:spring-webmvc:jar:5.2.13.RELEASE (compile) <img id="_img59" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep58', '_img59' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep58" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Web MVC</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Web MVC</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div>
+<ul>
+<li>org.springframework:spring-aop:jar:5.2.13.RELEASE (compile) <img id="_img61" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep60', '_img61' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep60" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring AOP</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring AOP</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li>
+<li>org.springframework:spring-context:jar:5.2.13.RELEASE (compile) <img id="_img63" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep62', '_img63' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep62" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Context</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Context</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li>
+<li>org.springframework:spring-expression:jar:5.2.13.RELEASE (compile) <img id="_img65" src="./images/icon_info_sml.gif" alt="[Information]" onclick="toggleDependencyDetails( '_dep64', '_img65' );" style="cursor: pointer; vertical-align: text-bottom;"></img><div id="_dep64" style="display:none">
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>Spring Expression Language (SpEL)</th></tr>
+<tr class="b">
+<td>
+<p><b>&#x63cf;&#x8ff0;: </b>Spring Expression Language (SpEL)</p>
+<p><b>URL: </b><a class="externalLink" href="https://github.com/spring-projects/spring-framework">https://github.com/spring-projects/spring-framework</a></p>
+<p><b>&#x9879;&#x76ee;&#x6388;&#x6743;: </b><a class="externalLink" href="https://www.apache.org/licenses/LICENSE-2.0">Apache License, Version 2.0</a></p></td></tr></table></div></li></ul></li></ul></li></ul></li></ul></section></section><a name="a.E6.89.A7.E7.85.A7"></a><section>
+<h2><a name="a.E6.89.A7.E7.85.A7"></a>&#x6267;&#x7167;</h2>
+<p><b>MIT License: </b>JUL to SLF4J bridge, SLF4J API Module</p>
+<p><b>Apache License, Version 2.0: </b>Apache Log4j API, Apache Log4j to SLF4J Adapter, SnakeYAML, Spring AOP, Spring Beans, Spring Commons Logging Bridge, Spring Context, Spring Core, Spring Expression Language (SpEL), Spring Web, Spring Web MVC, UtilsDemo, spring-boot, spring-boot-autoconfigure, spring-boot-starter, spring-boot-starter-json, spring-boot-starter-logging, spring-boot-starter-tomcat, spring-boot-starter-web, tomcat-embed-core, tomcat-embed-websocket</p>
+<p><b>EPL 2.0: </b>Jakarta Annotations API, Jakarta Expression Language 3.0</p>
+<p><b>GPL2 w/ CPE: </b>Jakarta Annotations API, Jakarta Expression Language 3.0</p>
+<p><b>GNU Lesser General Public License: </b>Logback Classic Module, Logback Core Module</p>
+<p><b>The Apache Software License, Version 2.0: </b>Jackson datatype: JSR310, Jackson datatype: jdk8, Jackson-annotations, Jackson-core, Jackson-module-parameter-names, jackson-databind</p>
+<p><b>Eclipse Public License - v 1.0: </b>Logback Classic Module, Logback Core Module</p></section><a name="a.E4.BE.9D.E8.B5.96.E6.96.87.E4.BB.B6.E8.AF.A6.E7.BB.86.E4.BF.A1.E6.81.AF"></a><section>
+<h2><a name="a.E4.BE.9D.E8.B5.96.E6.96.87.E4.BB.B6.E8.AF.A6.E7.BB.86.E4.BF.A1.E6.81.AF"></a>&#x4f9d;&#x8d56;&#x6587;&#x4ef6;&#x8be6;&#x7ec6;&#x4fe1;&#x606f;</h2>
+<table border="0" class="bodyTable">
+<tr class="a">
+<th>&#x6587;&#x4ef6;</th>
+<th>&#x5927;&#x5c0f;</th>
+<th>&#x6761;&#x76ee;</th>
+<th>&#x7c7b;</th>
+<th>&#x5305;</th>
+<th>Java Version</th>
+<th title="Indicates whether these dependencies have been compiled with debug information.">Debug Information</th></tr>
+<tr class="b">
+<td>logback-classic-1.2.3.jar</td>
+<td>290.3 kB</td>
+<td>234</td>
+<td>175</td>
+<td>27</td>
+<td>1.6</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>logback-core-1.2.3.jar</td>
+<td>471.9 kB</td>
+<td>419</td>
+<td>373</td>
+<td>36</td>
+<td>1.6</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>jackson-annotations-2.11.4.jar</td>
+<td>72.1 kB</td>
+<td>81</td>
+<td>69</td>
+<td>2</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>jackson-core-2.11.4.jar</td>
+<td>351.5 kB</td>
+<td>144</td>
+<td>118</td>
+<td>13</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>jackson-databind-2.11.4.jar</td>
+<td>1.4 MB</td>
+<td>693</td>
+<td>658</td>
+<td>22</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>jackson-datatype-jdk8-2.11.4.jar</td>
+<td>34.3 kB</td>
+<td>37</td>
+<td>23</td>
+<td>2</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>jackson-datatype-jsr310-2.11.4.jar</td>
+<td>111 kB</td>
+<td>76</td>
+<td>57</td>
+<td>6</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>jackson-module-parameter-names-2.11.4.jar</td>
+<td>9.3 kB</td>
+<td>19</td>
+<td>5</td>
+<td>2</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>jakarta.annotation-api-1.3.5.jar</td>
+<td>25.1 kB</td>
+<td>28</td>
+<td>15</td>
+<td>3</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>log4j-api-2.13.3.jar</td>
+<td>292.3 kB</td>
+<td>214</td>
+<td>180</td>
+<td>9</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>log4j-to-slf4j-2.13.3.jar</td>
+<td>17.5 kB</td>
+<td>22</td>
+<td>6</td>
+<td>1</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>tomcat-embed-core-9.0.43.jar</td>
+<td>3.4 MB</td>
+<td>1731</td>
+<td>1520</td>
+<td>79</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>tomcat-embed-websocket-9.0.43.jar</td>
+<td>271.8 kB</td>
+<td>205</td>
+<td>178</td>
+<td>6</td>
+<td>-</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>jakarta.el-3.0.3.jar</td>
+<td>237.8 kB</td>
+<td>182</td>
+<td>162</td>
+<td>6</td>
+<td>1.7</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>jul-to-slf4j-1.7.30.jar</td>
+<td>4.6 kB</td>
+<td>11</td>
+<td>1</td>
+<td>1</td>
+<td>1.5</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>slf4j-api-1.7.30.jar</td>
+<td>41.5 kB</td>
+<td>46</td>
+<td>34</td>
+<td>4</td>
+<td>1.5</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-aop-5.2.13.RELEASE.jar</td>
+<td>372.8 kB</td>
+<td>306</td>
+<td>277</td>
+<td>17</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>spring-beans-5.2.13.RELEASE.jar</td>
+<td>688.7 kB</td>
+<td>414</td>
+<td>384</td>
+<td>13</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-context-5.2.13.RELEASE.jar</td>
+<td>1.2 MB</td>
+<td>967</td>
+<td>880</td>
+<td>64</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>spring-core-5.2.13.RELEASE.jar</td>
+<td>1.4 MB</td>
+<td>1023</td>
+<td>965</td>
+<td>47</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-expression-5.2.13.RELEASE.jar</td>
+<td>282.2 kB</td>
+<td>164</td>
+<td>150</td>
+<td>6</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>spring-jcl-5.2.13.RELEASE.jar</td>
+<td>24 kB</td>
+<td>28</td>
+<td>17</td>
+<td>2</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-web-5.2.13.RELEASE.jar</td>
+<td>1.4 MB</td>
+<td>996</td>
+<td>926</td>
+<td>55</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>spring-webmvc-5.2.13.RELEASE.jar</td>
+<td>956.7 kB</td>
+<td>586</td>
+<td>541</td>
+<td>28</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-boot-2.3.9.RELEASE.jar</td>
+<td>1.1 MB</td>
+<td>837</td>
+<td>725</td>
+<td>71</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="a">
+<td>spring-boot-autoconfigure-2.3.9.RELEASE.jar</td>
+<td>1.5 MB</td>
+<td>1220</td>
+<td>1104</td>
+<td>99</td>
+<td>1.8</td>
+<td>Yes</td></tr>
+<tr class="b">
+<td>spring-boot-starter-2.3.9.RELEASE.jar</td>
+<td>4.8 kB</td>
+<td>4</td>
+<td>0</td>
+<td>0</td>
+<td>-</td>
+<td>No</td></tr>
+<tr class="a">
+<td>spring-boot-starter-json-2.3.9.RELEASE.jar</td>
+<td>4.7 kB</td>
+<td>4</td>
+<td>0</td>
+<td>0</td>
+<td>-</td>
+<td>No</td></tr>
+<tr class="b">
+<td>spring-boot-starter-logging-2.3.9.RELEASE.jar</td>
+<td>4.8 kB</td>
+<td>4</td>
+<td>0</td>
+<td>0</td>
+<td>-</td>
+<td>No</td></tr>
+<tr class="a">
+<td>spring-boot-starter-tomcat-2.3.9.RELEASE.jar</td>
+<td>4.8 kB</td>
+<td>4</td>
+<td>0</td>
+<td>0</td>
+<td>-</td>
+<td>No</td></tr>
+<tr class="b">
+<td>spring-boot-starter-web-2.3.9.RELEASE.jar</td>
+<td>4.8 kB</td>
+<td>4</td>
+<td>0</td>
+<td>0</td>
+<td>-</td>
+<td>No</td></tr>
+<tr class="a">
+<td>snakeyaml-1.26.jar</td>
+<td>309 kB</td>
+<td>253</td>
+<td>216</td>
+<td>20</td>
+<td>1.7</td>
+<td>Yes</td></tr>
+<tr class="b">
+<th>Total</th>
+<th>&#x5927;&#x5c0f;</th>
+<th>&#x6761;&#x76ee;</th>
+<th>&#x7c7b;</th>
+<th>&#x5305;</th>
+<th>Java Version</th>
+<th>Debug Information</th></tr>
+<tr class="a">
+<td>32</td>
+<td>16.4 MB</td>
+<td>10956</td>
+<td>9759</td>
+<td>641</td>
+<td>1.8</td>
+<td>27</td></tr>
+<tr class="b">
+<td>compile: 32</td>
+<td>compile: 16.4 MB</td>
+<td>compile: 10956</td>
+<td>compile: 9759</td>
+<td>compile: 641</td>
+<td>-</td>
+<td>compile: 27</td></tr></table></section>
+      </div>
+    </div>
+    <div class="clear">
+      <hr/>
+    </div>
+    <div id="footer">
+      <div class="xright">
+        Copyright &#169;      2021..      </div>
+      <div class="clear">
+        <hr/>
+      </div>
+    </div>
+  </body>
+</html>

BIN
target/site/images/close.gif


BIN
target/site/images/collapsed.gif


BIN
target/site/images/expanded.gif


BIN
target/site/images/external.png


BIN
target/site/images/icon_error_sml.gif


BIN
target/site/images/icon_info_sml.gif


BIN
target/site/images/icon_success_sml.gif


BIN
target/site/images/icon_warning_sml.gif


BIN
target/site/images/logos/build-by-maven-black.png


BIN
target/site/images/logos/build-by-maven-white.png


BIN
target/site/images/logos/maven-feather.png


BIN
target/site/images/newwindow.png