当前位置:   article > 正文

后端必知:遵循Google Java规范并引入checkstyle检查

后端必知:遵循Google Java规范并引入checkstyle检查

IDEA配置Code Style

协同工作时,为了保证统一的代码风格,要求我们使用的规范,如果大家都用的是 IDEA,则推荐使用 Google Code Style,推荐阅读Google Java 编程风格中文文档

先下载配置文件:github.com/google/styl…

以上是我的 code 配置,一般缩进就是 2个字符。

格式化代码快捷键,window上快捷键为(Ctrl + Alt + L),mac 上为 ⌥ + ⌘ + L。

Maven项目引入checkstyle检查

由于项目是 Maven 项目,希望在编译的时候自动执行检查,不需要额外手动执行,可以选择在 pom.xml 配置 maven-checkstyle-plugin插件,绑定到 Maven 的生命周期,这样在执行 mvn compile 等命令时自动触发执行检查。

引入依赖

一 maven 项目 pom 引入 checkstyle 依赖

多模块的 maven 项目,只需要在父模块的 pom.xml 里面配置插件即可,pom.xml 配置如下:

  1. <build>
  2. <plugins>
  3. <plugin>
  4. <groupId>org.springframework.boot</groupId>
  5. <artifactId>spring-boot-maven-plugin</artifactId>
  6. </plugin>
  7. <plugin>
  8. <groupId>org.apache.maven.plugins</groupId>
  9. <artifactId>maven-checkstyle-plugin</artifactId>
  10. <version>3.2.0</version>
  11. <configuration>
  12. <configLocation>checkstyle.xml</configLocation>
  13. <consoleOutput>true</consoleOutput>
  14. <failsOnError>true</failsOnError>
  15. <linkXRef>false</linkXRef>
  16. </configuration>
  17. <executions>
  18. <execution>
  19. <id>validate</id>
  20. <phase>validate</phase>
  21. <goals>
  22. <goal>check</goal>
  23. </goals>
  24. </execution>
  25. </executions>
  26. </plugin>
  27. </plugins>
  28. </build>
  29. 复制代码

checkstyle配置

二、下载 Google-checks.xml,下载地址:github.com/checkstyle/…

在项目中的位置如下所示:

下面是个人用的 checkstyle.xml 配置:

  1. <?xml version="1.0"?>
  2. <!DOCTYPE module PUBLIC
  3. "-//Checkstyle//DTD Checkstyle Configuration 1.3//EN"
  4. "https://checkstyle.org/dtds/configuration_1_3.dtd">
  5. <!--
  6. Checkstyle configuration that checks the Google coding conventions from Google Java Style
  7. that can be found at https://google.github.io/styleguide/javaguide.html
  8. Checkstyle is very configurable. Be sure to read the documentation at
  9. http://checkstyle.org (or in your downloaded distribution).
  10. To completely disable a check, just comment it out or delete it from the file.
  11. To suppress certain violations please review suppression filters.
  12. Authors: Max Vetrenko, Ruslan Diachenko, Roman Ivanov.
  13. -->
  14. <module name="Checker">
  15. <module name="SuppressWarningsFilter"/>
  16. <property name="charset" value="UTF-8"/>
  17. <!-- 检查出问题的错误级别,这里设置为 error,目的是便于 Maven 构建时,能在检查出问题后使构建失败. -->
  18. <property name="severity" value="error"/>
  19. <!-- 本 checkstyle 检查哪些文件. -->
  20. <property name="fileExtensions" value="java, properties, xml"/>
  21. <!-- Excludes all 'module-info.java' files -->
  22. <!-- See https://checkstyle.org/config_filefilters.html -->
  23. <module name="BeforeExecutionExclusionFileFilter">
  24. <property name="fileNamePattern" value="module\-info\.java$"/>
  25. </module>
  26. <!-- 配置抑制对某些文件进行检测的过滤器文件,默认是在 checkstyle-suppressions.xml 文件中. -->
  27. <!-- 具体如何进行配置可参考这里: https://checkstyle.org/config_filters.html#SuppressionFilter -->
  28. <module name="SuppressionFilter">
  29. <property name="file" value="${org.checkstyle.google.suppressionfilter.config}"
  30. default="checkstyle-suppressions.xml"/>
  31. <property name="optional" value="true"/>
  32. </module>
  33. <!-- Checks for whitespace -->
  34. <!-- 检查源代码中是否有制表符 (`tab`),默认检查每一行,
  35. 更多配置可参考这里: http://checkstyle.org/config_whitespace.html. -->
  36. <module name="FileTabCharacter">
  37. <property name="eachLine" value="true"/>
  38. </module>
  39. <!-- 列长度限制,这里我定义为 150,且对于含有 package, import, http 等关键字的忽略其列长度限制. -->
  40. <module name="LineLength">
  41. <property name="fileExtensions" value="java"/>
  42. <property name="max" value="150"/>
  43. <property name="ignorePattern"
  44. value="^package.*|^import.*|a href|href|http://|https://|ftp://"/>
  45. </module>
  46. <module name="TreeWalker">
  47. <module name="OuterTypeFilename"/>
  48. <!-- 检查非法的字符串标记,不使用八进制数和 Unicode 转义符. -->
  49. <module name="IllegalTokenText">
  50. <property name="tokens" value="STRING_LITERAL, CHAR_LITERAL"/>
  51. <property name="format"
  52. value="\\u00(09|0(a|A)|0(c|C)|0(d|D)|22|27|5(C|c))|\\(0(10|11|12|14|15|42|47)|134)"/>
  53. <property name="message"
  54. value="Consider using special escape sequence instead of octal value or Unicode escaped value."/>
  55. </module>
  56. <!-- 避免转义的 Unicode 字符,这里允许各种转义字符和对其进行尾部注释. -->
  57. <module name="AvoidEscapedUnicodeCharacters">
  58. <property name="allowEscapesForControlCharacters" value="true"/>
  59. <property name="allowByTailComment" value="true"/>
  60. <property name="allowNonPrintableEscapes" value="true"/>
  61. </module>
  62. <!-- 检查是否使用 '*' 号导入,是否只有一个顶级类,包语句和导入语句不需要换行的情况. -->
  63. <module name="AvoidStarImport"/>
  64. <module name="OneTopLevelClass"/>
  65. <module name="NoLineWrap">
  66. <property name="tokens" value="PACKAGE_DEF, IMPORT, STATIC_IMPORT"/>
  67. </module>
  68. <!-- 检查是否存在空语句块的情况,包括:while, do, for, if 等等. -->
  69. <module name="EmptyBlock">
  70. <property name="option" value="TEXT"/>
  71. <property name="tokens"
  72. value="LITERAL_TRY, LITERAL_FINALLY, LITERAL_IF, LITERAL_ELSE, LITERAL_SWITCH"/>
  73. </module>
  74. <!-- 检查需要双大括号 ('{', '}') 语句块的情况,包括:while, do, for, if, else. -->
  75. <module name="NeedBraces">
  76. <property name="tokens"
  77. value="LITERAL_DO, LITERAL_ELSE, LITERAL_FOR, LITERAL_IF, LITERAL_WHILE"/>
  78. </module>
  79. <!-- 检查左大括号 ('{') 及之前的代码须在一行的情况,包括:类定义, Lambda, for, finally, if 等等. -->
  80. <module name="LeftCurly">
  81. <property name="tokens"
  82. value="ANNOTATION_DEF, CLASS_DEF, CTOR_DEF, ENUM_CONSTANT_DEF, ENUM_DEF,
  83. INTERFACE_DEF, LAMBDA, LITERAL_CASE, LITERAL_CATCH, LITERAL_DEFAULT,
  84. LITERAL_DO, LITERAL_ELSE, LITERAL_FINALLY, LITERAL_FOR, LITERAL_IF,
  85. LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_TRY, LITERAL_WHILE, METHOD_DEF,
  86. OBJBLOCK, STATIC_INIT, RECORD_DEF, COMPACT_CTOR_DEF"/>
  87. </module>
  88. <!-- 检查右大括号 ('}') 与之后的代码在同一行情况,包括:try, catch, finally, if, else, do 等等. -->
  89. <module name="RightCurly">
  90. <property name="id" value="RightCurlySame"/>
  91. <property name="tokens"
  92. value="LITERAL_TRY, LITERAL_CATCH, LITERAL_FINALLY, LITERAL_IF, LITERAL_ELSE,
  93. LITERAL_DO"/>
  94. </module>
  95. <!-- 检查右大括号 ('}') 独立成行的情况,包括:类定义,方法定义,for, while, static 代码块等. -->
  96. <module name="RightCurly">
  97. <property name="id" value="RightCurlyAlone"/>
  98. <property name="option" value="alone"/>
  99. <property name="tokens"
  100. value="CLASS_DEF, METHOD_DEF, CTOR_DEF, LITERAL_FOR, LITERAL_WHILE, STATIC_INIT,
  101. INSTANCE_INIT, ANNOTATION_DEF, ENUM_DEF, INTERFACE_DEF, RECORD_DEF,
  102. COMPACT_CTOR_DEF"/>
  103. </module>
  104. <!-- 单个规则抑制过滤器,包含 XPath 元素和表达式的抑制。详情可参考这里:
  105. https://github.com/checkstyle/checkstyle/issues/7541 -->
  106. <module name="SuppressionXpathSingleFilter">
  107. <property name="id" value="RightCurlyAlone"/>
  108. <property name="query" value="//RCURLY[parent::SLIST[count(./*)=1]
  109. or preceding-sibling::*[last()][self::LCURLY]]"/>
  110. </module>
  111. <!-- 检查特殊符号后面是否有空格. -->
  112. <module name="WhitespaceAfter">
  113. <property name="tokens"
  114. value="COMMA, SEMI, TYPECAST, LITERAL_IF, LITERAL_ELSE,
  115. LITERAL_WHILE, LITERAL_DO, LITERAL_FOR, DO_WHILE"/>
  116. </module>
  117. <!-- 检查标识符周围是否有空白,空构造方法,空方法,空注解,空循环等的大括号中必须有空白,通常是空行. -->
  118. <module name="WhitespaceAround">
  119. <property name="allowEmptyConstructors" value="true"/>
  120. <property name="allowEmptyLambdas" value="true"/>
  121. <property name="allowEmptyMethods" value="true"/>
  122. <property name="allowEmptyTypes" value="true"/>
  123. <property name="allowEmptyLoops" value="true"/>
  124. <property name="ignoreEnhancedForColon" value="false"/>
  125. <property name="tokens"
  126. value="ASSIGN, BAND, BAND_ASSIGN, BOR, BOR_ASSIGN, BSR, BSR_ASSIGN, BXOR,
  127. BXOR_ASSIGN, COLON, DIV, DIV_ASSIGN, DO_WHILE, EQUAL, GE, GT, LAMBDA, LAND,
  128. LCURLY, LE, LITERAL_CATCH, LITERAL_DO, LITERAL_ELSE, LITERAL_FINALLY,
  129. LITERAL_FOR, LITERAL_IF, LITERAL_RETURN, LITERAL_SWITCH, LITERAL_SYNCHRONIZED,
  130. LITERAL_TRY, LITERAL_WHILE, LOR, LT, MINUS, MINUS_ASSIGN, MOD, MOD_ASSIGN,
  131. NOT_EQUAL, PLUS, PLUS_ASSIGN, QUESTION, RCURLY, SL, SLIST, SL_ASSIGN, SR,
  132. SR_ASSIGN, STAR, STAR_ASSIGN, LITERAL_ASSERT, TYPE_EXTENSION_AND"/>
  133. <message key="ws.notFollowed"
  134. value="WhitespaceAround: ''{0}'' is not followed by whitespace. Empty blocks
  135. may only be represented as '{}' when not part of a multi-block statement (4.1.3)"/>
  136. <message key="ws.notPreceded"
  137. value="WhitespaceAround: ''{0}'' is not preceded with whitespace."/>
  138. </module>
  139. <!-- 检查每行只能有一条语句和不能多变量声明. -->
  140. <module name="OneStatementPerLine"/>
  141. <module name="MultipleVariableDeclarations"/>
  142. <!-- 检查数组类型的风格,默认强制使用 Java 风格,即中括号紧跟在类型后面,如:'String[] arr'. -->
  143. <module name="ArrayTypeStyle"/>
  144. <!-- 检查 switch 是否有 default 分支,及“直落注释”的检查. -->
  145. <module name="MissingSwitchDefault"/>
  146. <module name="FallThrough"/>
  147. <!-- 检查 long 型变量值是否以大写的 'L' 结尾. -->
  148. <module name="UpperEll"/>
  149. <!-- 检查修饰符的顺序是否符合 Java 语言规范. 顺序应该是:'public protected private abstract default static final
  150. transient volatile synchronized native strictfp'. -->
  151. <module name="ModifierOrder"/>
  152. <!-- 检查主要分割模块之间是否有空行,包括:包语句,导入语句、静态导入、类定义、方法定义等等. -->
  153. <module name="EmptyLineSeparator">
  154. <property name="tokens"
  155. value="PACKAGE_DEF, IMPORT, STATIC_IMPORT, CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
  156. STATIC_INIT, INSTANCE_INIT, METHOD_DEF, CTOR_DEF, VARIABLE_DEF, RECORD_DEF,
  157. COMPACT_CTOR_DEF"/>
  158. <!-- 允许字段之间没有空行,不允许模块间或模块类有多个空行(即最多只有一个空行). -->
  159. <property name="allowNoEmptyLineBetweenFields" value="true"/>
  160. </module>
  161. <!-- 检查在英文点号 ('.') 处换行,且英文点号需在下一行的行首 (即:'nl'). -->
  162. <module name="SeparatorWrap">
  163. <property name="id" value="SeparatorWrapDot"/>
  164. <property name="tokens" value="DOT"/>
  165. <property name="option" value="nl"/>
  166. </module>
  167. <!-- 检查在英文逗号 (',') 处换行,且英文逗号需在上一行的行尾 (即:'eol'). -->
  168. <module name="SeparatorWrap">
  169. <property name="id" value="SeparatorWrapComma"/>
  170. <property name="tokens" value="COMMA"/>
  171. <property name="option" value="EOL"/>
  172. </module>
  173. <!-- 检查在省略号 ('...') 处换行,且省略号需在上一行的行尾 (即:'eol'). -->
  174. <module name="SeparatorWrap">
  175. <!-- ELLIPSIS is EOL until https://github.com/google/styleguide/issues/259 -->
  176. <property name="id" value="SeparatorWrapEllipsis"/>
  177. <property name="tokens" value="ELLIPSIS"/>
  178. <property name="option" value="EOL"/>
  179. </module>
  180. <!-- 检查在数组中括号 ('[]') 处换行,且中括号需在上一行的行尾 (即:'eol'). -->
  181. <module name="SeparatorWrap">
  182. <!-- ARRAY_DECLARATOR is EOL until https://github.com/google/styleguide/issues/258 -->
  183. <property name="id" value="SeparatorWrapArrayDeclarator"/>
  184. <property name="tokens" value="ARRAY_DECLARATOR"/>
  185. <property name="option" value="EOL"/>
  186. </module>
  187. <!-- 检查在方法引用双冒号 ('::') 处换行,且双冒号需在下一行的行首 (即:'nl'). -->
  188. <module name="SeparatorWrap">
  189. <property name="id" value="SeparatorWrapMethodRef"/>
  190. <property name="tokens" value="METHOD_REF"/>
  191. <property name="option" value="nl"/>
  192. </module>
  193. <!-- 检查包 (package) 名称是否符合 format 中定义的属性指定的格式. -->
  194. <module name="PackageName">
  195. <property name="format" value="^[a-z]+(\.[a-z][a-z0-9]*)*$"/>
  196. <message key="name.invalidPattern"
  197. value="Package name ''{0}'' must match pattern ''{1}''."/>
  198. </module>
  199. <!-- 检查类型名称(类、接口、枚举、注解等)是否符合大驼峰的格式. -->
  200. <module name="TypeName">
  201. <property name="tokens" value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
  202. ANNOTATION_DEF, RECORD_DEF"/>
  203. <message key="name.invalidPattern"
  204. value="Type name ''{0}'' must match pattern ''{1}''."/>
  205. </module>
  206. <!-- 检查类成员名称(属性、方法)是否符合 format 中定义的小驼峰格式. -->
  207. <module name="MemberName">
  208. <property name="format" value="^[a-z][a-z0-9][a-zA-Z0-9]*$"/>
  209. <message key="name.invalidPattern"
  210. value="Member name ''{0}'' must match pattern ''{1}''."/>
  211. </module>
  212. <!-- 检查参数名称是否符合 format 中定义的小驼峰格式. -->
  213. <module name="ParameterName">
  214. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
  215. <message key="name.invalidPattern"
  216. value="Parameter name ''{0}'' must match pattern ''{1}''."/>
  217. </module>
  218. <!-- 检查 Lambda 参数名称是否符合 format 中定义的小驼峰格式. -->
  219. <module name="LambdaParameterName">
  220. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
  221. <message key="name.invalidPattern"
  222. value="Lambda parameter name ''{0}'' must match pattern ''{1}''."/>
  223. </module>
  224. <!-- 检查 Catch 块中参数名称是否符合 format 中定义的小驼峰格式. -->
  225. <module name="CatchParameterName">
  226. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
  227. <message key="name.invalidPattern"
  228. value="Catch parameter name ''{0}'' must match pattern ''{1}''."/>
  229. </module>
  230. <!-- 检查局部变量名称是否符合 format 中定义的小驼峰格式. -->
  231. <module name="LocalVariableName">
  232. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
  233. <message key="name.invalidPattern"
  234. value="Local variable name ''{0}'' must match pattern ''{1}''."/>
  235. </module>
  236. <!-- 检查模式变量名称是否符合 format 中定义的小驼峰格式,可以参见 Java 14 中的 instanceOf 模式匹配功能. -->
  237. <module name="PatternVariableName">
  238. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
  239. <message key="name.invalidPattern"
  240. value="Pattern variable name ''{0}'' must match pattern ''{1}''."/>
  241. </module>
  242. <!-- 检查类中的泛型参数名称是否符合 format 中定义的格式. -->
  243. <module name="ClassTypeParameterName">
  244. <property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
  245. <message key="name.invalidPattern"
  246. value="Class type name ''{0}'' must match pattern ''{1}''."/>
  247. </module>
  248. <!-- 检查 Record 类中,组件名称是否符合 format 中定义的小驼峰格式. -->
  249. <module name="RecordComponentName">
  250. <property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$" />
  251. <message key="name.invalidPattern" value="Record component name ''{0}'' must match pattern ''{1}''." />
  252. </module>
  253. <!-- 检查 Record 泛型参数名称是否符合 format 中定义的格式. -->
  254. <module name="RecordTypeParameterName">
  255. <property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
  256. <message key="name.invalidPattern"
  257. value="Record type name ''{0}'' must match pattern ''{1}''."/>
  258. </module>
  259. <!-- 检查 Record 泛型参数名称是否符合 format 中定义的格式. -->
  260. <module name="RecordTypeParameterName">
  261. <property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
  262. <message key="name.invalidPattern"
  263. value="Record type name ''{0}'' must match pattern ''{1}''."/>
  264. </module>
  265. <!-- 检查方法中的的泛型参数名称是否符合 format 中定义的格式. -->
  266. <module name="MethodTypeParameterName">
  267. <property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
  268. <message key="name.invalidPattern"
  269. value="Method type name ''{0}'' must match pattern ''{1}''."/>
  270. </module>
  271. <!-- 检查接口中的的泛型参数名称是否符合 format 中定义的格式. -->
  272. <module name="InterfaceTypeParameterName">
  273. <property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
  274. <message key="name.invalidPattern"
  275. value="Interface type name ''{0}'' must match pattern ''{1}''."/>
  276. </module>
  277. <!-- 检查是否重写了 `Object.finalize` 方法(禁止重写). -->
  278. <module name="NoFinalizer"/>
  279. <!-- 检查尖括号 ('<', '>') 周围的空白,详情参考这里:
  280. https://checkstyle.sourceforge.io/config_whitespace.html#GenericWhitespace. -->
  281. <module name="GenericWhitespace">
  282. <message key="ws.followed"
  283. value="GenericWhitespace ''{0}'' is followed by whitespace."/>
  284. <message key="ws.preceded"
  285. value="GenericWhitespace ''{0}'' is preceded with whitespace."/>
  286. <message key="ws.illegalFollow"
  287. value="GenericWhitespace ''{0}'' should followed by whitespace."/>
  288. <message key="ws.notPreceded"
  289. value="GenericWhitespace ''{0}'' is not preceded with whitespace."/>
  290. </module>
  291. <!-- 检查缩进的空格数是否正确,基础缩进是 2 格,抛出异常和换行是 4 格. -->
  292. <module name="Indentation">
  293. <property name="basicOffset" value="2"/>
  294. <property name="braceAdjustment" value="0"/>
  295. <property name="caseIndent" value="2"/>
  296. <property name="throwsIndent" value="4"/>
  297. <property name="lineWrappingIndentation" value="4"/>
  298. <property name="arrayInitIndent" value="2"/>
  299. </module>
  300. <!-- 检查标识符名称中的缩写(连续大写字母)长度,自定义了一些可以连续大写字母的单词,如:XML, ID, VO 等等. -->
  301. <module name="AbbreviationAsWordInName">
  302. <!--忽略常量中的连续大写字母-->
  303. <property name="ignoreFinal" value="false"/>
  304. <property name="allowedAbbreviationLength" value="0"/>
  305. <property name="allowedAbbreviations"
  306. value="XML, URL, ID, FTP, DTO, VO, DO, TO, PO, BO, POJO, HTTP, IP, IE"/>
  307. <property name="tokens"
  308. value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, ANNOTATION_DEF, ANNOTATION_FIELD_DEF,
  309. PARAMETER_DEF, VARIABLE_DEF, METHOD_DEF, PATTERN_VARIABLE_DEF, RECORD_DEF,
  310. RECORD_COMPONENT_DEF"/>
  311. </module>
  312. <!-- 检查 switch 块中冒号之前是否没有空格. -->
  313. <module name="NoWhitespaceBeforeCaseDefaultColon"/>
  314. <module name="OverloadMethodsDeclarationOrder"/>
  315. <!-- 检查变量声明与其首次使用之间的距离,默认大于 3 行,如果该变量是 final 修饰的,默认会忽略此规则. -->
  316. <module name="VariableDeclarationUsageDistance"/>
  317. <!-- 检查自定义导入 (import) 顺序,按 ASCII 码顺序对导入进行分组排序. -->
  318. <module name="CustomImportOrder">
  319. <property name="sortImportsInGroupAlphabetically" value="true"/>
  320. <property name="separateLineBetweenGroups" value="true"/>
  321. <property name="customImportOrderRules" value="STATIC###THIRD_PARTY_PACKAGE"/>
  322. <property name="tokens" value="IMPORT, STATIC_IMPORT, PACKAGE_DEF"/>
  323. </module>
  324. <!-- 检查方法定义,方法调用等情况与参数列表左括号之间无空格,在标识符与左括号之间也不允许换行(默认). -->
  325. <module name="MethodParamPad">
  326. <property name="tokens"
  327. value="CTOR_DEF, LITERAL_NEW, METHOD_CALL, METHOD_DEF,
  328. SUPER_CTOR_CALL, ENUM_CONSTANT_DEF, RECORD_DEF"/>
  329. </module>
  330. <!-- 检查标记符之前没有空白. -->
  331. <module name="NoWhitespaceBefore">
  332. <property name="tokens"
  333. value="COMMA, SEMI, POST_INC, POST_DEC, DOT,
  334. LABELED_STAT, METHOD_REF"/>
  335. <property name="allowLineBreaks" value="true"/>
  336. </module>
  337. <!-- 检查小括号 ('(', ')') 的空白填充策略,默认小括号内的两边没有空格. -->
  338. <module name="ParenPad">
  339. <property name="tokens"
  340. value="ANNOTATION, ANNOTATION_FIELD_DEF, CTOR_CALL, CTOR_DEF, DOT, ENUM_CONSTANT_DEF,
  341. EXPR, LITERAL_CATCH, LITERAL_DO, LITERAL_FOR, LITERAL_IF, LITERAL_NEW,
  342. LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_WHILE, METHOD_CALL,
  343. METHOD_DEF, QUESTION, RESOURCE_SPECIFICATION, SUPER_CTOR_CALL, LAMBDA,
  344. RECORD_DEF"/>
  345. </module>
  346. <!-- 检查在运算符处换行的策略,且换行后运算符在下一行的行首,并定义了常见的运算符. -->
  347. <module name="OperatorWrap">
  348. <property name="option" value="NL"/>
  349. <property name="tokens"
  350. value="BAND, BOR, BSR, BXOR, DIV, EQUAL, GE, GT, LAND, LE, LITERAL_INSTANCEOF, LOR,
  351. LT, MINUS, MOD, NOT_EQUAL, PLUS, QUESTION, SL, SR, STAR, METHOD_REF,
  352. TYPE_EXTENSION_AND "/>
  353. </module>
  354. <!-- 检查注解的标注位置,该类注解必须独立成行,包括:类,接口,枚举,方法等处的注解定义. -->
  355. <module name="AnnotationLocation">
  356. <property name="id" value="AnnotationLocationMostCases"/>
  357. <property name="tokens"
  358. value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, METHOD_DEF, CTOR_DEF,
  359. RECORD_DEF, COMPACT_CTOR_DEF"/>
  360. </module>
  361. <!-- 检查局部变量注解的标注位置,该类注解允许在同一行. -->
  362. <module name="AnnotationLocation">
  363. <property name="id" value="AnnotationLocationVariables"/>
  364. <property name="tokens" value="VARIABLE_DEF"/>
  365. <property name="allowSamelineMultipleAnnotations" value="true"/>
  366. </module>
  367. <module name="NonEmptyAtclauseDescription"/>
  368. <!-- 检查 Javadoc 是否位于正确的位置,其会检查无效的 Javadoc. -->
  369. <module name="InvalidJavadocPosition"/>
  370. <!-- 检查 Javadoc 中 '@' 注解标签的缩进,默认换行时会缩进 4 格,此处自定义为0格. -->
  371. <module name="JavadocTagContinuationIndentation">
  372. <property name="offset" value="0"/>
  373. </module>
  374. <module name="SummaryJavadoc">
  375. <property name="forbiddenSummaryFragments"
  376. value="^@return the *|^This method returns |^A [{]@code [a-zA-Z0-9]+[}]( is a )"/>
  377. </module>
  378. <!-- 检查 Javadoc 中的段落. -->
  379. <module name="JavadocParagraph"/>
  380. <!-- 使用空行来隔开 '@Param''@return' 等块标签组. -->
  381. <module name="RequireEmptyLineBeforeBlockTagGroup"/>
  382. <!-- 约定 Javadoc 注解标记的先后顺序. -->
  383. <module name="AtclauseOrder">
  384. <!--<property name="tagOrder"
  385. value="@param, @return, @throws, @deprecated, @author, @博客, @网站, @date, @description"/>-->
  386. <property name="tagOrder" value="@param, @return, @throws, @deprecated, @author"/>
  387. <property name="target"
  388. value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, METHOD_DEF, CTOR_DEF, VARIABLE_DEF"/>
  389. </module>
  390. <!-- Javadoc 方法注释,检查 protected 及其之上的 protected 域,且不能缺失 @param 和 @return 的注解标记. -->
  391. <module name="JavadocMethod">
  392. <property name="accessModifiers" value="public"/>
  393. <property name="allowMissingParamTags" value="true"/>
  394. <property name="allowMissingReturnTag" value="true"/>
  395. <!-- Javadoc 方法注释,允许在具有 @Override 和 @Test 注解的方法上写 Javadoc 注释. -->
  396. <property name="allowedAnnotations" value="Override, Test"/>
  397. <property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF, COMPACT_CTOR_DEF"/>
  398. </module>
  399. <!-- 可以缺失的 Javadoc 方法注释,检查 protected 域,且最小代码行数小于 0 行时,才可缺失(即无论如何都不可缺失). -->
  400. <!--<module name="MissingJavadocMethod">
  401. <property name="scope" value="public"/>
  402. <property name="minLineCount" value="0"/>
  403. <property name="allowedAnnotations" value="Override, Test"/>
  404. <property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF,
  405. COMPACT_CTOR_DEF"/>
  406. </module> -->
  407. <!-- 检查类、枚举、接口和注解接口定义中缺少的 Javadoc 注释. -->
  408. <!--<module name="MissingJavadocType">
  409. <property name="scope" value="protected"/>
  410. <property name="tokens"
  411. value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
  412. RECORD_DEF, ANNOTATION_DEF"/>
  413. <property name="excludeScope" value="nothing"/>
  414. </module>-->
  415. <!-- 方法名使用小驼峰的模式. -->
  416. <module name="MethodName">
  417. <property name="format" value="^[a-z][a-z0-9][a-zA-Z0-9_]*$"/>
  418. <message key="name.invalidPattern"
  419. value="Method name ''{0}'' must match pattern ''{1}''."/>
  420. </module>
  421. <!-- 单行的 Javadoc,不忽略行内标签. -->
  422. <module name="SingleLineJavadoc">
  423. <property name="ignoreInlineTags" value="false"/>
  424. </module>
  425. <!-- 检查允许空 catch 块的情况,通过将异常实例变量名命名为 'expected', 就会跳过该项的检查. -->
  426. <module name="EmptyCatchBlock">
  427. <property name="exceptionVariableName" value="expected"/>
  428. </module>
  429. <!-- 检查注释和周围代码之间的缩进.-->
  430. <module name="CommentsIndentation">
  431. <property name="tokens" value="SINGLE_LINE_COMMENT, BLOCK_COMMENT_BEGIN"/>
  432. </module>
  433. <!-- 规则抑制过滤器,包含 XPath 元素和表达式的抑制。详情可参考这里:
  434. https://checkstyle.org/config_filters.html#SuppressionXpathFilter -->
  435. <module name="SuppressionXpathFilter">
  436. <property name="file" value="${org.checkstyle.google.suppressionxpathfilter.config}"
  437. default="checkstyle-xpath-suppressions.xml"/>
  438. <property name="optional" value="true"/>
  439. </module>
  440. <module name="SuppressWarningsHolder"/>
  441. <!--抑制注释过滤器,以CHECKSTYLE.OFF开始,CHECKSTYLE.ON结束-->
  442. <module name="SuppressionCommentFilter">
  443. <property name="offCommentFormat" value="CHECKSTYLE.OFF\: ([\w\|]+)"/>
  444. <property name="onCommentFormat" value="CHECKSTYLE.ON\: ([\w\|]+)"/>
  445. <property name="checkFormat" value="$1"/>
  446. </module>
  447. <module name="SuppressWithNearbyCommentFilter">
  448. <property name="commentFormat" value="CHECKSTYLE.SUPPRESS\: ([\w\|]+)"/>
  449. <!-- $1 refers to the first match group in the regex defined in commentFormat -->
  450. <property name="checkFormat" value="$1"/>
  451. <!-- The check is suppressed in the next line of code after the comment -->
  452. <property name="influenceFormat" value="1"/>
  453. </module>
  454. </module>
  455. </module>
  456. 复制代码

执行 checkstyle 检查

执行命令

  1. mvn checkstyle:check
  2. //或者
  3. mvn checkstyle:checkstyle
  4. 复制代码

或者点击如下命令:

SuppressionFilter

SuppressionFilter 可以抑制以 Treewalker 或 Checker 作为父模块的 Checks。

比如说我们的代码如下所示:

  1. private boolean isContainChinese(String text) {
  2. Pattern p = Pattern
  3. .compile("[\u4E00-\u9FA5!,。()《》“”?:;【】]");
  4. Matcher m = p.matcher(text);
  5. return m.find();
  6. }
  7. 复制代码

被检测出这样一个错误:

  1. StringUtils.java:102:18: 不要使用Unicode转义字符。 [AvoidEscapedUnicodeCharacters]
  2. 复制代码

如何避免错误被检测呢,可以这样设置:

  1. <module name="SuppressionFilter">
  2. <property name="file" value="suppressions.xml"/>
  3. <property name="optional" value="true"/>
  4. </module>
  5. 复制代码

同时在 checkstyle.xml 所在目前下新建 suppressions.xml,内容如下:

  1. <?xml version="1.0"?>
  2. <!DOCTYPE suppressions PUBLIC
  3. "-//Checkstyle//DTD SuppressionFilter Configuration 1.2//EN"
  4. "https://checkstyle.org/dtds/suppressions_1_2.dtd">
  5. <suppressions>
  6. <suppress checks="AvoidEscapedUnicodeCharacters"
  7. files="StringUtils.java"
  8. lines="101-103"/>
  9. <suppress message="不要使用Unicode转义字符"/>
  10. </suppressions>
  11. 复制代码

引入checkStyle插件

三、idea 中引入checkStyle插件

如果每次想针对单个文件执行 checkstyle 检查,可以这样做:

1、在IDEA中安装checstyle插件:

2、新增自定义的checkstyle.xml文件:

关于 suppressions 需要设置属性值,其实 checkstyle.xml 文件中已经有默认值了,但还需要设置,不然最后会报错。

最后效果如下:

3、随便进入一个文件,然后右键选择 checkstyle

执行效果如下:

重要检查及应对措施

关于 checkstyle 所有的检查通知,具体有哪些提示信息,参考这篇文章。这里贴一点提示信息:

  1. indentation.error.multi=''{0}'' 缩进了{1}个缩进符,应为以下缩进之一:{2}。
  2. indentation.child.error.multi=''{0}'' 子元素进了{1}个缩进符,应为以下缩进之一:{2}。
  3. indentation.error=''{0}'' 缩进了{1}个缩进符,应为{2}个。
  4. indentation.child.error=''{0}'' 子元素缩进了{1}个缩进符,应为{2}个。
  5. comments.indentation.single=注释应与第{0}行代码同样缩进{2}个缩进符,而不是{1}个。
  6. comments.indentation.block=注释应与第{0}行代码同样缩进{2}个缩进符,而不是{1}个。
  7. 复制代码

1、首先是方法注释,如果要求检查每个方法上是否有注释,则添加如下配置(默认自带)。位于 JavadocMethod model下,这点结合项目实际情况,按需决定是否需要保留。

比如说检查结果为:

  1. BaseRedisConfig.java:30:3: 缺少 Javadoc 。 [MissingJavadocMethod]
  2. 复制代码

对应代码为:

  1. @Bean
  2. public RedisTemplate<String, Object> redisTemplate(
  3. RedisConnectionFactory redisConnectionFactory) {
  4. RedisSerializer<Object> serializer = redisSerializer();
  5. RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
  6. redisTemplate.setConnectionFactory(redisConnectionFactory);
  7. redisTemplate.setKeySerializer(new StringRedisSerializer());
  8. redisTemplate.setValueSerializer(serializer);
  9. redisTemplate.setHashKeySerializer(new StringRedisSerializer());
  10. redisTemplate.setHashValueSerializer(serializer);
  11. redisTemplate.afterPropertiesSet();
  12. return redisTemplate;
  13. }
  14. 复制代码

提示我们上述方法没有注释,如果想避开该检查,则删除如下配置。

  1. <module name="MissingJavadocMethod">
  2. <property name="scope" value="public"/>
  3. <property name="minLineCount" value="2"/>
  4. <property name="allowedAnnotations" value="Override, Test"/>
  5. <property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF,
  6. COMPACT_CTOR_DEF"/>
  7. </module>
  8. <module name="MissingJavadocType">
  9. <property name="scope" value="protected"/>
  10. <property name="tokens"
  11. value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
  12. RECORD_DEF, ANNOTATION_DEF"/>
  13. <property name="excludeScope" value="nothing"/>
  14. </module>
  15. 复制代码

如果要求增加方法注释,则需要注意如下事项:

  1. 要包含方法说明
  2. 要包含参数列表
  3. 要包含返回值类型
  4. 要严格注意注释的缩进和星号数量

2、类注释

代码如下所示:

  1. /**
  2. * @author hresh
  3. * @博客 https://juejin.cn/user/2664871918047063
  4. * @网站 https://www.hreshhao.com/
  5. * @date 2022/9/14 9:07 下午
  6. * @description
  7. */
  8. @Configuration
  9. public class SpringDocConfig{
  10. }
  11. 复制代码

检查后提示如下信息:

  1. SpringDocConfig.java:10: 缺少摘要javadoc[SummaryJavadoc]
  2. 复制代码

归根结底是因为类文件没有描述信息,@description 是我们自定义的标签,并不会 checkStyle 所认可,默认类文件描述信息要在第一行,且必须要有结尾标识,即添加英文句号结尾,如下所示:

  1. /**
  2. * 测试.
  3. *
  4. * @author hresh
  5. * @博客 https://juejin.cn/user/2664871918047063
  6. * @网站 https://www.hreshhao.com/
  7. * @date 2022/9/14 9:07 下午
  8. */
  9. 复制代码

这就要求我们修改类文件注释模版,如下所示:

  1. /**
  2. *
  3. *
  4. * @author hresh
  5. * @博客 https://juejin.cn/user/2664871918047063
  6. * @网站 https://www.hreshhao.com/
  7. * @date ${DATE} ${TIME}
  8. */
  9. 复制代码

3、类注释缩进符不对

代码如下所示:

  1. /**
  2. * @author hresh
  3. * @博客 https://juejin.cn/user/2664871918047063
  4. * @网站 https://www.hreshhao.com/
  5. * @date 2022/9/14 9:07 下午
  6. * @description
  7. */
  8. @Configuration
  9. public class SpringDocConfig{
  10. }
  11. 复制代码

检查结果为:

  1. SpringDocConfig.java:12: Javadoc 缩进级别错误,应为 4 个缩进符。 [JavadocTagContinuationIndentation]
  2. SpringDocConfig.java:13: Javadoc 缩进级别错误,应为 4 个缩进符。 [JavadocTagContinuationIndentation]
  3. 复制代码

如何解决呢?最简单的方法就是去除该配置,除此之外,还需要修改类文件注释模版,如下是我自定义的模版:

  1. /**
  2. * @author hresh
  3. * @博客 https://juejin.cn/user/2664871918047063
  4. * @网站 https://www.hreshhao.com/
  5. * @date ${DATE} ${TIME}
  6. * @description
  7. */
  8. 复制代码

但仍然报错,经过测试发现,如果改成这样就不会检查报错了。

  1. /*
  2. * @author hresh
  3. * @博客 https://juejin.cn/user/2664871918047063
  4. * @网站 https://www.hreshhao.com/
  5. * @date ${DATE} ${TIME}
  6. * @description
  7. */
  8. 复制代码

经过查询 /* /和 /* */的区别,发现后者是说明注释,可以添加到 javadoc 中,后续可以生成 HTML 格式的代码报告。

最佳解决方案为:

  1. <!-- 检查 Javadoc 中 '@' 注解标签的缩进,默认换行时会缩进 4 格,此处自定义为0格. -->
  2. <module name="JavadocTagContinuationIndentation">
  3. <property name="offset" value="0"/>
  4. </module>
  5. 复制代码

如何添加 property,以及 name 值是什么,这点需要参考官方文档: checkstyle.org/config_java…

4、方法参数名称

代码示例:

  1. public static <T> PageResult<T> ok(IPage<T> iPage)
  2. 复制代码

检查报错

  1. Parameter name 'iPage' must match pattern '^[a-z]([a-z0-9][a-zA-Z0-9]*)?$'. [ParameterName]
  2. 复制代码

5、方法注释

代码示例:

  1. /**
  2. * 驼峰转下划线
  3. *
  4. * @param name
  5. * @return
  6. */
  7. public static String camelToUnderscore(String name)
  8. 复制代码

上述代码会有两个报错:

  1. StringUtils.java:15: Javadoc的第一句缺少一个结束时期。 [SummaryJavadoc]
  2. StringUtils.java:18: @标签应有非空说明。 [NonEmptyAtclauseDescription]
  3. 复制代码

解决方案:

  1. /**
  2. * 驼峰转下划线.
  3. *
  4. * @param name 字符串
  5. * @return
  6. */
  7. public static String camelToUnderscore(String name)
  8. 复制代码

实际应用

当我们使用 orm-generator 脚手架生成模版代码,并复制到项目中后,格式可能存在问题,比如这种错误:

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/201822
推荐阅读
相关标签
  

闽ICP备14008679号