遵循 Google Java 规范并引入 Checkstyle 检查,代码写的像诗一样!
Posted Java知音_
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了遵循 Google Java 规范并引入 Checkstyle 检查,代码写的像诗一样!相关的知识,希望对你有一定的参考价值。
“你的代码规范是什么?如何保证同一个项目中协同工作的人书写出来的代码格式一致?”
虽然我们知道一些代码规范,比如说“对于重复出现的代码进行封装复用,写好注释等等“,但这些规范我们不可能要求每个开发人员牢牢记住,那么就需要引入一些约定俗成的配置,来帮助我们对代码进行检查,及时发现问题并解决问题。
IDEA配置Code Style
协同工作时,为了保证统一的代码风格,要求我们使用的规范,如果大家都用的是 IDEA,则推荐使用 Google Code Style,推荐阅读Google Java 编程风格中文文档:
https://github.com/fantasticmao/google-java-style-guide-zh_cn
先下载配置文件:
https://github.com/google/styleguide/blob/gh-pages/intellij-java-google-style.xml
以上是我的 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 配置如下:
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-checkstyle-plugin</artifactId>
<version>3.2.0</version>
<configuration>
<configLocation>checkstyle.xml</configLocation>
<consoleOutput>true</consoleOutput>
<failsOnError>true</failsOnError>
<linkXRef>false</linkXRef>
</configuration>
<executions>
<execution>
<id>validate</id>
<phase>validate</phase>
<goals>
<goal>check</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
checkstyle配置
二、下载 Google-checks.xml,下载地址:
https://github.com/checkstyle/checkstyle/blob/master/src/main/resources/google_checks.xml
在项目中的位置如下所示:
下面是个人用的 checkstyle.xml 配置(给大家参考):
<?xml version="1.0"?>
<!DOCTYPE module PUBLIC
"-//Checkstyle//DTD Checkstyle Configuration 1.3//EN"
"https://checkstyle.org/dtds/configuration_1_3.dtd">
<!--
Checkstyle configuration that checks the Google coding conventions from Google Java Style
that can be found at https://google.github.io/styleguide/javaguide.html
Checkstyle is very configurable. Be sure to read the documentation at
http://checkstyle.org (or in your downloaded distribution).
To completely disable a check, just comment it out or delete it from the file.
To suppress certain violations please review suppression filters.
Authors: Max Vetrenko, Ruslan Diachenko, Roman Ivanov.
-->
<module name="Checker">
<module name="SuppressWarningsFilter"/>
<property name="charset" value="UTF-8"/>
<!-- 检查出问题的错误级别,这里设置为 error,目的是便于 Maven 构建时,能在检查出问题后使构建失败. -->
<property name="severity" value="error"/>
<!-- 本 checkstyle 检查哪些文件. -->
<property name="fileExtensions" value="java, properties, xml"/>
<!-- Excludes all 'module-info.java' files -->
<!-- See https://checkstyle.org/config_filefilters.html -->
<module name="BeforeExecutionExclusionFileFilter">
<property name="fileNamePattern" value="module\\-info\\.java$"/>
</module>
<!-- 配置抑制对某些文件进行检测的过滤器文件,默认是在 checkstyle-suppressions.xml 文件中. -->
<!-- 具体如何进行配置可参考这里: https://checkstyle.org/config_filters.html#SuppressionFilter -->
<module name="SuppressionFilter">
<property name="file" value="$org.checkstyle.google.suppressionfilter.config"
default="checkstyle-suppressions.xml"/>
<property name="optional" value="true"/>
</module>
<!-- Checks for whitespace -->
<!-- 检查源代码中是否有制表符 (`tab`),默认检查每一行,
更多配置可参考这里: http://checkstyle.org/config_whitespace.html. -->
<module name="FileTabCharacter">
<property name="eachLine" value="true"/>
</module>
<!-- 列长度限制,这里我定义为 150,且对于含有 package, import, http 等关键字的忽略其列长度限制. -->
<module name="LineLength">
<property name="fileExtensions" value="java"/>
<property name="max" value="150"/>
<property name="ignorePattern"
value="^package.*|^import.*|a href|href|http://|https://|ftp://"/>
</module>
<module name="TreeWalker">
<module name="OuterTypeFilename"/>
<!-- 检查非法的字符串标记,不使用八进制数和 Unicode 转义符. -->
<module name="IllegalTokenText">
<property name="tokens" value="STRING_LITERAL, CHAR_LITERAL"/>
<property name="format"
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)"/>
<property name="message"
value="Consider using special escape sequence instead of octal value or Unicode escaped value."/>
</module>
<!-- 避免转义的 Unicode 字符,这里允许各种转义字符和对其进行尾部注释. -->
<module name="AvoidEscapedUnicodeCharacters">
<property name="allowEscapesForControlCharacters" value="true"/>
<property name="allowByTailComment" value="true"/>
<property name="allowNonPrintableEscapes" value="true"/>
</module>
<!-- 检查是否使用 '*' 号导入,是否只有一个顶级类,包语句和导入语句不需要换行的情况. -->
<module name="AvoidStarImport"/>
<module name="OneTopLevelClass"/>
<module name="NoLineWrap">
<property name="tokens" value="PACKAGE_DEF, IMPORT, STATIC_IMPORT"/>
</module>
<!-- 检查是否存在空语句块的情况,包括:while, do, for, if 等等. -->
<module name="EmptyBlock">
<property name="option" value="TEXT"/>
<property name="tokens"
value="LITERAL_TRY, LITERAL_FINALLY, LITERAL_IF, LITERAL_ELSE, LITERAL_SWITCH"/>
</module>
<!-- 检查需要双大括号 ('', '') 语句块的情况,包括:while, do, for, if, else. -->
<module name="NeedBraces">
<property name="tokens"
value="LITERAL_DO, LITERAL_ELSE, LITERAL_FOR, LITERAL_IF, LITERAL_WHILE"/>
</module>
<!-- 检查左大括号 ('') 及之前的代码须在一行的情况,包括:类定义, Lambda, for, finally, if 等等. -->
<module name="LeftCurly">
<property name="tokens"
value="ANNOTATION_DEF, CLASS_DEF, CTOR_DEF, ENUM_CONSTANT_DEF, ENUM_DEF,
INTERFACE_DEF, LAMBDA, LITERAL_CASE, LITERAL_CATCH, LITERAL_DEFAULT,
LITERAL_DO, LITERAL_ELSE, LITERAL_FINALLY, LITERAL_FOR, LITERAL_IF,
LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_TRY, LITERAL_WHILE, METHOD_DEF,
OBJBLOCK, STATIC_INIT, RECORD_DEF, COMPACT_CTOR_DEF"/>
</module>
<!-- 检查右大括号 ('') 与之后的代码在同一行情况,包括:try, catch, finally, if, else, do 等等. -->
<module name="RightCurly">
<property name="id" value="RightCurlySame"/>
<property name="tokens"
value="LITERAL_TRY, LITERAL_CATCH, LITERAL_FINALLY, LITERAL_IF, LITERAL_ELSE,
LITERAL_DO"/>
</module>
<!-- 检查右大括号 ('') 独立成行的情况,包括:类定义,方法定义,for, while, static 代码块等. -->
<module name="RightCurly">
<property name="id" value="RightCurlyAlone"/>
<property name="option" value="alone"/>
<property name="tokens"
value="CLASS_DEF, METHOD_DEF, CTOR_DEF, LITERAL_FOR, LITERAL_WHILE, STATIC_INIT,
INSTANCE_INIT, ANNOTATION_DEF, ENUM_DEF, INTERFACE_DEF, RECORD_DEF,
COMPACT_CTOR_DEF"/>
</module>
<!-- 单个规则抑制过滤器,包含 XPath 元素和表达式的抑制。详情可参考这里:
https://github.com/checkstyle/checkstyle/issues/7541 -->
<module name="SuppressionXpathSingleFilter">
<property name="id" value="RightCurlyAlone"/>
<property name="query" value="//RCURLY[parent::SLIST[count(./*)=1]
or preceding-sibling::*[last()][self::LCURLY]]"/>
</module>
<!-- 检查特殊符号后面是否有空格. -->
<module name="WhitespaceAfter">
<property name="tokens"
value="COMMA, SEMI, TYPECAST, LITERAL_IF, LITERAL_ELSE,
LITERAL_WHILE, LITERAL_DO, LITERAL_FOR, DO_WHILE"/>
</module>
<!-- 检查标识符周围是否有空白,空构造方法,空方法,空注解,空循环等的大括号中必须有空白,通常是空行. -->
<module name="WhitespaceAround">
<property name="allowEmptyConstructors" value="true"/>
<property name="allowEmptyLambdas" value="true"/>
<property name="allowEmptyMethods" value="true"/>
<property name="allowEmptyTypes" value="true"/>
<property name="allowEmptyLoops" value="true"/>
<property name="ignoreEnhancedForColon" value="false"/>
<property name="tokens"
value="ASSIGN, BAND, BAND_ASSIGN, BOR, BOR_ASSIGN, BSR, BSR_ASSIGN, BXOR,
BXOR_ASSIGN, COLON, DIV, DIV_ASSIGN, DO_WHILE, EQUAL, GE, GT, LAMBDA, LAND,
LCURLY, LE, LITERAL_CATCH, LITERAL_DO, LITERAL_ELSE, LITERAL_FINALLY,
LITERAL_FOR, LITERAL_IF, LITERAL_RETURN, LITERAL_SWITCH, LITERAL_SYNCHRONIZED,
LITERAL_TRY, LITERAL_WHILE, LOR, LT, MINUS, MINUS_ASSIGN, MOD, MOD_ASSIGN,
NOT_EQUAL, PLUS, PLUS_ASSIGN, QUESTION, RCURLY, SL, SLIST, SL_ASSIGN, SR,
SR_ASSIGN, STAR, STAR_ASSIGN, LITERAL_ASSERT, TYPE_EXTENSION_AND"/>
<message key="ws.notFollowed"
value="WhitespaceAround: ''0'' is not followed by whitespace. Empty blocks
may only be represented as '' when not part of a multi-block statement (4.1.3)"/>
<message key="ws.notPreceded"
value="WhitespaceAround: ''0'' is not preceded with whitespace."/>
</module>
<!-- 检查每行只能有一条语句和不能多变量声明. -->
<module name="OneStatementPerLine"/>
<module name="MultipleVariableDeclarations"/>
<!-- 检查数组类型的风格,默认强制使用 Java 风格,即中括号紧跟在类型后面,如:'String[] arr'. -->
<module name="ArrayTypeStyle"/>
<!-- 检查 switch 是否有 default 分支,及“直落注释”的检查. -->
<module name="MissingSwitchDefault"/>
<module name="FallThrough"/>
<!-- 检查 long 型变量值是否以大写的 'L' 结尾. -->
<module name="UpperEll"/>
<!-- 检查修饰符的顺序是否符合 Java 语言规范. 顺序应该是:'public protected private abstract default static final
transient volatile synchronized native strictfp'. -->
<module name="ModifierOrder"/>
<!-- 检查主要分割模块之间是否有空行,包括:包语句,导入语句、静态导入、类定义、方法定义等等. -->
<module name="EmptyLineSeparator">
<property name="tokens"
value="PACKAGE_DEF, IMPORT, STATIC_IMPORT, CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
STATIC_INIT, INSTANCE_INIT, METHOD_DEF, CTOR_DEF, VARIABLE_DEF, RECORD_DEF,
COMPACT_CTOR_DEF"/>
<!-- 允许字段之间没有空行,不允许模块间或模块类有多个空行(即最多只有一个空行). -->
<property name="allowNoEmptyLineBetweenFields" value="true"/>
</module>
<!-- 检查在英文点号 ('.') 处换行,且英文点号需在下一行的行首 (即:'nl'). -->
<module name="SeparatorWrap">
<property name="id" value="SeparatorWrapDot"/>
<property name="tokens" value="DOT"/>
<property name="option" value="nl"/>
</module>
<!-- 检查在英文逗号 (',') 处换行,且英文逗号需在上一行的行尾 (即:'eol'). -->
<module name="SeparatorWrap">
<property name="id" value="SeparatorWrapComma"/>
<property name="tokens" value="COMMA"/>
<property name="option" value="EOL"/>
</module>
<!-- 检查在省略号 ('...') 处换行,且省略号需在上一行的行尾 (即:'eol'). -->
<module name="SeparatorWrap">
<!-- ELLIPSIS is EOL until https://github.com/google/styleguide/issues/259 -->
<property name="id" value="SeparatorWrapEllipsis"/>
<property name="tokens" value="ELLIPSIS"/>
<property name="option" value="EOL"/>
</module>
<!-- 检查在数组中括号 ('[]') 处换行,且中括号需在上一行的行尾 (即:'eol'). -->
<module name="SeparatorWrap">
<!-- ARRAY_DECLARATOR is EOL until https://github.com/google/styleguide/issues/258 -->
<property name="id" value="SeparatorWrapArrayDeclarator"/>
<property name="tokens" value="ARRAY_DECLARATOR"/>
<property name="option" value="EOL"/>
</module>
<!-- 检查在方法引用双冒号 ('::') 处换行,且双冒号需在下一行的行首 (即:'nl'). -->
<module name="SeparatorWrap">
<property name="id" value="SeparatorWrapMethodRef"/>
<property name="tokens" value="METHOD_REF"/>
<property name="option" value="nl"/>
</module>
<!-- 检查包 (package) 名称是否符合 format 中定义的属性指定的格式. -->
<module name="PackageName">
<property name="format" value="^[a-z]+(\\.[a-z][a-z0-9]*)*$"/>
<message key="name.invalidPattern"
value="Package name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查类型名称(类、接口、枚举、注解等)是否符合大驼峰的格式. -->
<module name="TypeName">
<property name="tokens" value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
ANNOTATION_DEF, RECORD_DEF"/>
<message key="name.invalidPattern"
value="Type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查类成员名称(属性、方法)是否符合 format 中定义的小驼峰格式. -->
<module name="MemberName">
<property name="format" value="^[a-z][a-z0-9][a-zA-Z0-9]*$"/>
<message key="name.invalidPattern"
value="Member name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查参数名称是否符合 format 中定义的小驼峰格式. -->
<module name="ParameterName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
<message key="name.invalidPattern"
value="Parameter name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查 Lambda 参数名称是否符合 format 中定义的小驼峰格式. -->
<module name="LambdaParameterName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
<message key="name.invalidPattern"
value="Lambda parameter name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查 Catch 块中参数名称是否符合 format 中定义的小驼峰格式. -->
<module name="CatchParameterName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
<message key="name.invalidPattern"
value="Catch parameter name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查局部变量名称是否符合 format 中定义的小驼峰格式. -->
<module name="LocalVariableName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
<message key="name.invalidPattern"
value="Local variable name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查模式变量名称是否符合 format 中定义的小驼峰格式,可以参见 Java 14 中的 instanceOf 模式匹配功能. -->
<module name="PatternVariableName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$"/>
<message key="name.invalidPattern"
value="Pattern variable name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查类中的泛型参数名称是否符合 format 中定义的格式. -->
<module name="ClassTypeParameterName">
<property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
<message key="name.invalidPattern"
value="Class type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查 Record 类中,组件名称是否符合 format 中定义的小驼峰格式. -->
<module name="RecordComponentName">
<property name="format" value="^[a-z]([a-z0-9][a-zA-Z0-9]*)?$" />
<message key="name.invalidPattern" value="Record component name ''0'' must match pattern ''1''." />
</module>
<!-- 检查 Record 泛型参数名称是否符合 format 中定义的格式. -->
<module name="RecordTypeParameterName">
<property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
<message key="name.invalidPattern"
value="Record type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查 Record 泛型参数名称是否符合 format 中定义的格式. -->
<module name="RecordTypeParameterName">
<property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
<message key="name.invalidPattern"
value="Record type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查方法中的的泛型参数名称是否符合 format 中定义的格式. -->
<module name="MethodTypeParameterName">
<property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
<message key="name.invalidPattern"
value="Method type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查接口中的的泛型参数名称是否符合 format 中定义的格式. -->
<module name="InterfaceTypeParameterName">
<property name="format" value="(^[A-Z][0-9]?)$|([A-Z][a-zA-Z0-9]*[T]$)"/>
<message key="name.invalidPattern"
value="Interface type name ''0'' must match pattern ''1''."/>
</module>
<!-- 检查是否重写了 `Object.finalize` 方法(禁止重写). -->
<module name="NoFinalizer"/>
<!-- 检查尖括号 ('<', '>') 周围的空白,详情参考这里:
https://checkstyle.sourceforge.io/config_whitespace.html#GenericWhitespace. -->
<module name="GenericWhitespace">
<message key="ws.followed"
value="GenericWhitespace ''0'' is followed by whitespace."/>
<message key="ws.preceded"
value="GenericWhitespace ''0'' is preceded with whitespace."/>
<message key="ws.illegalFollow"
value="GenericWhitespace ''0'' should followed by whitespace."/>
<message key="ws.notPreceded"
value="GenericWhitespace ''0'' is not preceded with whitespace."/>
</module>
<!-- 检查缩进的空格数是否正确,基础缩进是 2 格,抛出异常和换行是 4 格. -->
<module name="Indentation">
<property name="basicOffset" value="2"/>
<property name="braceAdjustment" value="0"/>
<property name="caseIndent" value="2"/>
<property name="throwsIndent" value="4"/>
<property name="lineWrappingIndentation" value="4"/>
<property name="arrayInitIndent" value="2"/>
</module>
<!-- 检查标识符名称中的缩写(连续大写字母)长度,自定义了一些可以连续大写字母的单词,如:XML, ID, VO 等等. -->
<module name="AbbreviationAsWordInName">
<!--忽略常量中的连续大写字母-->
<property name="ignoreFinal" value="false"/>
<property name="allowedAbbreviationLength" value="0"/>
<property name="allowedAbbreviations"
value="XML, URL, ID, FTP, DTO, VO, DO, TO, PO, BO, POJO, HTTP, IP, IE"/>
<property name="tokens"
value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, ANNOTATION_DEF, ANNOTATION_FIELD_DEF,
PARAMETER_DEF, VARIABLE_DEF, METHOD_DEF, PATTERN_VARIABLE_DEF, RECORD_DEF,
RECORD_COMPONENT_DEF"/>
</module>
<!-- 检查 switch 块中冒号之前是否没有空格. -->
<module name="NoWhitespaceBeforeCaseDefaultColon"/>
<module name="OverloadMethodsDeclarationOrder"/>
<!-- 检查变量声明与其首次使用之间的距离,默认大于 3 行,如果该变量是 final 修饰的,默认会忽略此规则. -->
<module name="VariableDeclarationUsageDistance"/>
<!-- 检查自定义导入 (import) 顺序,按 ASCII 码顺序对导入进行分组排序. -->
<module name="CustomImportOrder">
<property name="sortImportsInGroupAlphabetically" value="true"/>
<property name="separateLineBetweenGroups" value="true"/>
<property name="customImportOrderRules" value="STATIC###THIRD_PARTY_PACKAGE"/>
<property name="tokens" value="IMPORT, STATIC_IMPORT, PACKAGE_DEF"/>
</module>
<!-- 检查方法定义,方法调用等情况与参数列表左括号之间无空格,在标识符与左括号之间也不允许换行(默认). -->
<module name="MethodParamPad">
<property name="tokens"
value="CTOR_DEF, LITERAL_NEW, METHOD_CALL, METHOD_DEF,
SUPER_CTOR_CALL, ENUM_CONSTANT_DEF, RECORD_DEF"/>
</module>
<!-- 检查标记符之前没有空白. -->
<module name="NoWhitespaceBefore">
<property name="tokens"
value="COMMA, SEMI, POST_INC, POST_DEC, DOT,
LABELED_STAT, METHOD_REF"/>
<property name="allowLineBreaks" value="true"/>
</module>
<!-- 检查小括号 ('(', ')') 的空白填充策略,默认小括号内的两边没有空格. -->
<module name="ParenPad">
<property name="tokens"
value="ANNOTATION, ANNOTATION_FIELD_DEF, CTOR_CALL, CTOR_DEF, DOT, ENUM_CONSTANT_DEF,
EXPR, LITERAL_CATCH, LITERAL_DO, LITERAL_FOR, LITERAL_IF, LITERAL_NEW,
LITERAL_SWITCH, LITERAL_SYNCHRONIZED, LITERAL_WHILE, METHOD_CALL,
METHOD_DEF, QUESTION, RESOURCE_SPECIFICATION, SUPER_CTOR_CALL, LAMBDA,
RECORD_DEF"/>
</module>
<!-- 检查在运算符处换行的策略,且换行后运算符在下一行的行首,并定义了常见的运算符. -->
<module name="OperatorWrap">
<property name="option" value="NL"/>
<property name="tokens"
value="BAND, BOR, BSR, BXOR, DIV, EQUAL, GE, GT, LAND, LE, LITERAL_INSTANCEOF, LOR,
LT, MINUS, MOD, NOT_EQUAL, PLUS, QUESTION, SL, SR, STAR, METHOD_REF,
TYPE_EXTENSION_AND "/>
</module>
<!-- 检查注解的标注位置,该类注解必须独立成行,包括:类,接口,枚举,方法等处的注解定义. -->
<module name="AnnotationLocation">
<property name="id" value="AnnotationLocationMostCases"/>
<property name="tokens"
value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, METHOD_DEF, CTOR_DEF,
RECORD_DEF, COMPACT_CTOR_DEF"/>
</module>
<!-- 检查局部变量注解的标注位置,该类注解允许在同一行. -->
<module name="AnnotationLocation">
<property name="id" value="AnnotationLocationVariables"/>
<property name="tokens" value="VARIABLE_DEF"/>
<property name="allowSamelineMultipleAnnotations" value="true"/>
</module>
<module name="NonEmptyAtclauseDescription"/>
<!-- 检查 Javadoc 是否位于正确的位置,其会检查无效的 Javadoc. -->
<module name="InvalidJavadocPosition"/>
<!-- 检查 Javadoc 中 '@' 注解标签的缩进,默认换行时会缩进 4 格,此处自定义为0格. -->
<module name="JavadocTagContinuationIndentation">
<property name="offset" value="0"/>
</module>
<module name="SummaryJavadoc">
<property name="forbiddenSummaryFragments"
value="^@return the *|^This method returns |^A []@code [a-zA-Z0-9]+[]( is a )"/>
</module>
<!-- 检查 Javadoc 中的段落. -->
<module name="JavadocParagraph"/>
<!-- 使用空行来隔开 '@Param'、'@return' 等块标签组. -->
<module name="RequireEmptyLineBeforeBlockTagGroup"/>
<!-- 约定 Javadoc 注解标记的先后顺序. -->
<module name="AtclauseOrder">
<!--<property name="tagOrder"
value="@param, @return, @throws, @deprecated, @author, @博客, @网站, @date, @description"/>-->
<property name="tagOrder" value="@param, @return, @throws, @deprecated, @author"/>
<property name="target"
value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF, METHOD_DEF, CTOR_DEF, VARIABLE_DEF"/>
</module>
<!-- Javadoc 方法注释,检查 protected 及其之上的 protected 域,且不能缺失 @param 和 @return 的注解标记. -->
<module name="JavadocMethod">
<property name="accessModifiers" value="public"/>
<property name="allowMissingParamTags" value="true"/>
<property name="allowMissingReturnTag" value="true"/>
<!-- Javadoc 方法注释,允许在具有 @Override 和 @Test 注解的方法上写 Javadoc 注释. -->
<property name="allowedAnnotations" value="Override, Test"/>
<property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF, COMPACT_CTOR_DEF"/>
</module>
<!-- 可以缺失的 Javadoc 方法注释,检查 protected 域,且最小代码行数小于 0 行时,才可缺失(即无论如何都不可缺失). -->
<!--<module name="MissingJavadocMethod">
<property name="scope" value="public"/>
<property name="minLineCount" value="0"/>
<property name="allowedAnnotations" value="Override, Test"/>
<property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF,
COMPACT_CTOR_DEF"/>
</module> -->
<!-- 检查类、枚举、接口和注解接口定义中缺少的 Javadoc 注释. -->
<!--<module name="MissingJavadocType">
<property name="scope" value="protected"/>
<property name="tokens"
value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
RECORD_DEF, ANNOTATION_DEF"/>
<property name="excludeScope" value="nothing"/>
</module>-->
<!-- 方法名使用小驼峰的模式. -->
<module name="MethodName">
<property name="format" value="^[a-z][a-z0-9][a-zA-Z0-9_]*$"/>
<message key="name.invalidPattern"
value="Method name ''0'' must match pattern ''1''."/>
</module>
<!-- 单行的 Javadoc,不忽略行内标签. -->
<module name="SingleLineJavadoc">
<property name="ignoreInlineTags" value="false"/>
</module>
<!-- 检查允许空 catch 块的情况,通过将异常实例变量名命名为 'expected', 就会跳过该项的检查. -->
<module name="EmptyCatchBlock">
<property name="exceptionVariableName" value="expected"/>
</module>
<!-- 检查注释和周围代码之间的缩进.-->
<module name="CommentsIndentation">
<property name="tokens" value="SINGLE_LINE_COMMENT, BLOCK_COMMENT_BEGIN"/>
</module>
<!-- 规则抑制过滤器,包含 XPath 元素和表达式的抑制。详情可参考这里:
https://checkstyle.org/config_filters.html#SuppressionXpathFilter -->
<module name="SuppressionXpathFilter">
<property name="file" value="$org.checkstyle.google.suppressionxpathfilter.config"
default="checkstyle-xpath-suppressions.xml"/>
<property name="optional" value="true"/>
</module>
<module name="SuppressWarningsHolder"/>
<!--抑制注释过滤器,以CHECKSTYLE.OFF开始,CHECKSTYLE.ON结束-->
<module name="SuppressionCommentFilter">
<property name="offCommentFormat" value="CHECKSTYLE.OFF\\: ([\\w\\|]+)"/>
<property name="onCommentFormat" value="CHECKSTYLE.ON\\: ([\\w\\|]+)"/>
<property name="checkFormat" value="$1"/>
</module>
<module name="SuppressWithNearbyCommentFilter">
<property name="commentFormat" value="CHECKSTYLE.SUPPRESS\\: ([\\w\\|]+)"/>
<!-- $1 refers to the first match group in the regex defined in commentFormat -->
<property name="checkFormat" value="$1"/>
<!-- The check is suppressed in the next line of code after the comment -->
<property name="influenceFormat" value="1"/>
</module>
</module>
</module>
执行 checkstyle 检查
执行命令
mvn checkstyle:check
//或者
mvn checkstyle:checkstyle
或者点击如下命令:
SuppressionFilter
SuppressionFilter
可以抑制以 Treewalker
或 Checker 作为父模块的 Checks。
比如说我们的代码如下所示:
private boolean isContainChinese(String text)
Pattern p = Pattern
.compile("[\\u4E00-\\u9FA5!,。()《》“”?:;【】]");
Matcher m = p.matcher(text);
return m.find();
被检测出这样一个错误:
StringUtils.java:102:18: 不要使用Unicode转义字符。 [AvoidEscapedUnicodeCharacters]
如何避免错误被检测呢,可以这样设置:
<module name="SuppressionFilter">
<property name="file" value="suppressions.xml"/>
<property name="optional" value="true"/>
</module>
同时在 checkstyle.xml 所在目前下新建 suppressions.xml,内容如下:
<?xml version="1.0"?>
<!DOCTYPE suppressions PUBLIC
"-//Checkstyle//DTD SuppressionFilter Configuration 1.2//EN"
"https://checkstyle.org/dtds/suppressions_1_2.dtd">
<suppressions>
<suppress checks="AvoidEscapedUnicodeCharacters"
files="StringUtils.java"
lines="101-103"/>
<suppress message="不要使用Unicode转义字符"/>
</suppressions>
引入checkStyle插件
idea 中引入checkStyle插件
如果每次想针对单个文件执行 checkstyle 检查,可以这样做:
1、在IDEA中安装checstyle插件:
2、新增自定义的checkstyle.xml文件:
关于 suppressions 需要设置属性值,其实 checkstyle.xml 文件中已经有默认值了,但还需要设置,不然最后会报错。
最后效果如下:
3、随便进入一个文件,然后右键选择 checkstyle
执行效果如下:
重要检查及应对措施
关于 checkstyle 所有的检查通知,具体有哪些提示信息,参考这篇:
https://github.com/checkstyle/checkstyle/issues/3001
这里贴一点提示信息:
indentation.error.multi=''0'' 缩进了1个缩进符,应为以下缩进之一:2。
indentation.child.error.multi=''0'' 子元素进了1个缩进符,应为以下缩进之一:2。
indentation.error=''0'' 缩进了1个缩进符,应为2个。
indentation.child.error=''0'' 子元素缩进了1个缩进符,应为2个。
comments.indentation.single=注释应与第0行代码同样缩进2个缩进符,而不是1个。
comments.indentation.block=注释应与第0行代码同样缩进2个缩进符,而不是1个。
1、首先是方法注释,如果要求检查每个方法上是否有注释,则添加如下配置(默认自带)。位于 JavadocMethod model
下,这点结合项目实际情况,按需决定是否需要保留。
比如说检查结果为:
BaseRedisConfig.java:30:3: 缺少 Javadoc 。 [MissingJavadocMethod]
对应代码为:
@Bean
public RedisTemplate<String, Object> redisTemplate(
RedisConnectionFactory redisConnectionFactory)
RedisSerializer<Object> serializer = redisSerializer();
RedisTemplate<String, Object> redisTemplate = new RedisTemplate<>();
redisTemplate.setConnectionFactory(redisConnectionFactory);
redisTemplate.setKeySerializer(new StringRedisSerializer());
redisTemplate.setValueSerializer(serializer);
redisTemplate.setHashKeySerializer(new StringRedisSerializer());
redisTemplate.setHashValueSerializer(serializer);
redisTemplate.afterPropertiesSet();
return redisTemplate;
提示我们上述方法没有注释,如果想避开该检查,则删除如下配置。
<module name="MissingJavadocMethod">
<property name="scope" value="public"/>
<property name="minLineCount" value="2"/>
<property name="allowedAnnotations" value="Override, Test"/>
<property name="tokens" value="METHOD_DEF, CTOR_DEF, ANNOTATION_FIELD_DEF,
COMPACT_CTOR_DEF"/>
</module>
<module name="MissingJavadocType">
<property name="scope" value="protected"/>
<property name="tokens"
value="CLASS_DEF, INTERFACE_DEF, ENUM_DEF,
RECORD_DEF, ANNOTATION_DEF"/>
<property name="excludeScope" value="nothing"/>
</module>
如果要求增加方法注释,则需要注意如下事项:
要包含方法说明
要包含参数列表
要包含返回值类型
要严格注意注释的缩进和星号数量
2、类注释
代码如下所示:
/**
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date 2022/9/14 9:07 下午
* @description
*/
@Configuration
public class SpringDocConfig
检查后提示如下信息:
SpringDocConfig.java:10: 缺少摘要javadoc。 [SummaryJavadoc]
归根结底是因为类文件没有描述信息,@description 是我们自定义的标签,并不会 checkStyle 所认可,默认类文件描述信息要在第一行,且必须要有结尾标识,即添加英文句号结尾,如下所示:
/**
* 测试.
*
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date 2022/9/14 9:07 下午
*/
这就要求我们修改类文件注释模版,如下所示:
/**
*
*
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date $DATE $TIME
*/
3、类注释缩进符不对
代码如下所示:
/**
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date 2022/9/14 9:07 下午
* @description
*/
@Configuration
public class SpringDocConfig
检查结果为:
SpringDocConfig.java:12: Javadoc 缩进级别错误,应为 4 个缩进符。 [JavadocTagContinuationIndentation]
SpringDocConfig.java:13: Javadoc 缩进级别错误,应为 4 个缩进符。 [JavadocTagContinuationIndentation]
如何解决呢?最简单的方法就是去除该配置,除此之外,还需要修改类文件注释模版,如下是我自定义的模版:
/**
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date $DATE $TIME
* @description
*/
但仍然报错,经过测试发现,如果改成这样就不会检查报错了。
/*
* @author hresh
* @博客 https://juejin.cn/user/2664871918047063
* @网站 https://www.hreshhao.com/
* @date $DATE $TIME
* @description
*/
经过查询 /* /
和 /* */
的区别,发现后者是说明注释,可以添加到 javadoc 中,后续可以生成 HTML 格式的代码报告。
最佳解决方案为:
<!-- 检查 Javadoc 中 '@' 注解标签的缩进,默认换行时会缩进 4 格,此处自定义为0格. -->
<module name="JavadocTagContinuationIndentation">
<property name="offset" value="0"/>
</module>
如何添加 property,以及 name 值是什么,这点需要参考官方文档:
https://checkstyle.org/config_javadoc.html#JavadocTagContinuationIndentation
4、方法参数名称
代码示例:
public static <T> PageResult<T> ok(IPage<T> iPage)
检查报错
Parameter name 'iPage' must match pattern '^[a-z]([a-z0-9][a-zA-Z0-9]*)?$'. [ParameterName]
5、方法注释
代码示例:
/**
* 驼峰转下划线
*
* @param name
* @return
*/
public static String camelToUnderscore(String name)
上述代码会有两个报错:
StringUtils.java:15: Javadoc的第一句缺少一个结束时期。 [SummaryJavadoc]
StringUtils.java:18: @标签应有非空说明。 [NonEmptyAtclauseDescription]
解决方案:
/**
* 驼峰转下划线.
*
* @param name 字符串
* @return
*/
public static String camelToUnderscore(String name)
实际应用
当我们使用 orm-generator 脚手架生成模版代码,并复制到项目中后,格式可能存在问题,比如这种错误:
参考文献
Google Java 代码规范[1]
idea java项目引入checkstyle检查[2]
Maven配置checkstyle,SpotBugs以及Jacoco插件[3]
参考资料
[1]
https://leer.moe/2017/09/18/google-java-style-guide/: https://link.juejin.cn?target=https%3A%2F%2Fleer.moe%2F2017%2F09%2F18%2Fgoogle-java-style-guide%2F
[2]https://www.cnblogs.com/songchaolin/p/12714215.html: https://link.juejin.cn?target=https%3A%2F%2Fwww.cnblogs.com%2Fsongchaolin%2Fp%2F12714215.html
[3]https://juejin.cn/post/6905364481413480461#heading-0: https://juejin.cn/post/6905364481413480461#heading-0
作者:hresh
来源:juejin.cn/post/7178683426852044858
推荐
PS:因为公众号平台更改了推送规则,如果不想错过内容,记得读完点一下“在看”,加个“星标”,这样每次新文章推送才会第一时间出现在你的订阅列表里。点“在看”支持我们吧!
以上是关于遵循 Google Java 规范并引入 Checkstyle 检查,代码写的像诗一样!的主要内容,如果未能解决你的问题,请参考以下文章