介绍

在我们的Java应用程序中复制/粘贴代码通常不好,但是有时这是不可避免的。 例如,项目License3j在Feature类中为其支持的每种XXX类型提供了一个isXXX方法。 在这种情况下,我们要做的就是写

 public boolean isBinary() { 
         return type == Type.BINARY; 
     } 
     public boolean isString() { 
         return type == Type.STRING; 
     } 
     public boolean isByte() { 
         return type == Type.BYTE; 
     } 
     public boolean isShort() { 
         return type == Type.SHORT; 
     }  and so on 

应用程序支持的每种功能类型。 那里有一些类型:Binary,String,Byte,Short,Int,Long,Float,Double,BigInteger,BigDecimal,Date,UUID。 键入所有非常相似的方法不仅是无聊的任务,而且容易出错。 很少有人擅长执行此类重复性任务。 为了避免这种情况,我们可以使用Java :: Geci框架,作为最简单的解决方案,我们可以使用生成器Iterate。

POM依赖

要使用生成器,我们必须添加依赖项

 < dependency > 
     < groupId >com.javax0.geci</ groupId > 
     < artifactId >javageci-core</ artifactId > 
     < scope >test</ scope > 
     < version >1.4.0</ version >  </ dependency > 

该库仅在测试运行时执行,因此使用它并不意味着任何额外的依赖关系。 谁想要使用许可证3j库都不需要使用Java :: Geci。 这只是test范围中使用的开发工具。

单元测试运行

依赖关系不会自行运行。 毕竟依赖不是程序。 它是打包到JAR中的一堆类文件,可在类路径上使用。 我们必须执行生成器,并且必须通过创建单元测试的框架来完成:

 @Test 
     @DisplayName ( "run Iterate on the sources" ) 
     void runIterate() throws IOException { 
         Geci geci = new Geci(); 
         Assertions.assertFalse( 
             geci.register(Iterate.builder() 
                               .define(ctx -> ctx.segment().param( "TYPE" , ctx.segment().getParam( "Type" ).orElse( "" ).toUpperCase())) 
                               .build()) 
                 .generate() 
             , geci.failed() 
         ); 
     } 

它创建一个Geci对象,使用生成器实例Geci ,然后在配置的框架Geci对象上调用generate() 。 就目前而言, define()调用似乎有点神秘。 稍后我们将阐明这一点。

源代码准备

执行构建之前的最后一步是定义模板以及要插入模板的值。 无需编写所有方法,而是编写模板和编辑器折叠段:

 /* TEMPLATE 
     LOOP Type=Binary|String|Byte|Short|Int|Long|Float|Double|BigInteger|BigDecimal|Date|UUID 
     public boolean is{{Type}}() { 
         return type == Type.{{TYPE}}; 
     } 
      */ 
     //<editor-fold id="iterate"> 
     //</editor-fold> 

当我们通过框架执行生成器时,它将为占位符Type每个值评估模板,并将每个{{Type}}替换为实际值。 生成的代码将插入id “ iterate”的编辑器折叠段中。

查看模板,您会发现有一个占位符{{TYPE}} ,未在列表中定义。 这是统一测试define()进入图片的地方。 它定义一个使用上下文的使用者,并使用该上下文读取Type的实际值,创建该值的大写版本并将其分配给名为TYPE的段参数。

通常就是这样。 使用生成器还有其他功能,例如每次迭代定义多个值分配给不同的占位符,转义或跳过行等。 关于这些内容的摘录是您可以阅读文档的摘录,您可以阅读最新且完整的内容https://github.com/verhas/javageci/blob/master/ITERATE.adoc

文档摘录

在要使用生成器的Java源文件中,必须使用注释@Geci("iterate")来注释该类。
您也可以改用@Iterate批注,该批注在
javageci-core-annotations模块。 这将指示Geci框架您要在给定类中使用iterate生成器。

TEMPLATE

模板在/\*TEMPLATETEMPLATE行之后开始。
/*和单词之间前后可以有空格
TEMPLATE但线上不应有其他任何东西。 当生成器看到这样的行时,它将开始收集以下行作为模板的内容。

模板的末尾由一行上带有*/信号线表示,没有其他内容(空格除外)。

模板的内容可以包含{{}}之间的参数
字符,与胡子模板程序使用的字符类似。
(生成器不使用胡须,模板处理更为简单。)

LOOP

在收集模板的行时,某些行被识别为模板的参数定义。 这些行不会进入模板的主干。 (这些行上的命令名称始终为大写。)

如您在简介中所见

 LOOP type =int|long|short 

不是模板文本的一部分。 它通过类型指示发生器迭代并设置参数{{type}}在文本到int第一, long第二和short的最后。 这样,您可以遍历单个参数的多个值。

更复杂的模板可能需要多个参数。 在这种情况下,您可以将它们在LOOP行中列出为

 LOOP type ,var=int,aInt|long,aLong|short,aShort 

这将告诉生成器将参数设置{{type}}相同的方式,前三个迭代,但同时也设置了参数{{var}}aInt在第一循环中,以aLong在第二循环中和最后一个循环中的aShort

如果值列表太长,则可以将列表分成多条LOOP行。 但是,在这种情况下,必须在第二行,第三行等等在LOOP行中重复这些变量。
它们的顺序可能有所不同,但是如果某些LOOP行中存在未定义的变量,则将解析引用该变量的占位符并将其保留为{{placeholder}}形式。

上面的例子也可以写成

 LOOP type ,var=int,aInt 
     LOOP var, type =aLong,long 
     LOOP type ,var=short,aShort 

并得出与上述LOOP相同的值:

 LOOP type ,var=int,aInt|long,aLong|short,aShort 

默认

从文件的开头到结尾处理模板,并按此顺序准备生成的代码。
生成的代码的内容将直接插入模板editor-foldeditor-fold部分。 虽然这样的id
editor-fold段并不是很有趣,您必须为每个段指定一个唯一的id 。 这是Java :: Geci框架的限制。

进阶使用

EDITOR-FOLD-ID

您可能会遇到多个模板,这些模板遍历不同的值,并且您希望结果进入相同的editor-fold
分割。 可以使用EDITOR_FOLD_ID

在下面的例子中

 package javax0.geci.iterate.sutclasses;  public class IterateOverMultipleValues { 
     /* TEMPLATE 
     {{ type }} get_{{ type }}Value(){ 
       {{ type }} {{variable}} = 0; 
       return {{variable}}; 
     } 
     LOOP type ,variable=int,i|long,l|short,s 
     EDITOR-FOLD-ID getters 
      */ 
     // 
             // nothing gets here 
     // 
     // 
     int get_intValue(){ 
       int i = 0; 
       return i; 
     } 
     long get_longValue(){ 
       long l = 0; 
       return l; 
     } 
     short get_shortValue(){ 
       short s = 0; 
       return s; 
     } 
     //  } 

生成的代码进入具有id名称的editor-fold
getters即使这不是遵循模板定义的getters

使用此功能可以将生成的代码从多个迭代模板发送到单个段中。 通常,将模板和段保持在一起是一个好习惯。

ESCAPESKIP

模板的结尾由*/ 。 这实质上是评论的结尾。 如果要在模板中包含注释(如JavaDoc),会发生什么情况。 您可以在注释行的末尾添加*/字符,但其中仍包含一些字符。 该解决方案不是很好,它实际上是一种解决方法。

要有一条完全是注释结尾的行,或者是模板处理可以解释的任何行,例如LOOP行,您应该在前一行中包含除ESCAPE任何行。 这将告诉模板处理将下一行包括在模板文本中,并在此后的行继续进行正常处理。

同样,您可以使用SKIP行来完全忽略下一行。
使用这两个命令,您可以将任何内容包括到模板中。

一个示例显示如何将JavaDoc注释包括到模板中:

 package javax0.geci.iterate.sutclasses;  public class SkippedLines { 
     /* TEMPLATE 
     /** 
      * A simple zero getter serving as a test example 
      * @ return zero in the type {{ type }} 
     ESCAPE 
      */ 
     // SKIP 
     /* 
     {{ type }} get_{{ type }}Value(){ 
       {{ type }} {{variable}} = 0; 
       return {{variable}}; 
     } 
     LOOP type ,variable=int,i|long,l|short,s 
     EDITOR-FOLD-ID getters 
      */ 
     // 
     /** 
      * A simple zero getter serving as a test example 
      * @ return zero in the type int 
      */ 
     int get_intValue(){ 
       int i = 0; 
       return i; 
     } 
     /** 
      * A simple zero getter serving as a test example 
      * @ return zero in the type long 
      */ 
     long get_longValue(){ 
       long l = 0; 
       return l; 
     } 
     /** 
      * A simple zero getter serving as a test example 
      * @ return zero in the type short 
      */ 
     short get_shortValue(){ 
       short s = 0; 
       return s; 
     } 
     //  } 

模板以注释开头,并且注释实际上可以包含任何其他开头的注释。 Java注释不嵌套。 模板的末尾是包含*/字符串的行。 我们希望该行成为模板的一部分,因此我们在该行之前
ESCAPE因此它不会被解释为模板的末尾。 另一方面,对于Java,这将结束注释。 要继续模板,我们必须“返回”注释模式,因为我们不希望Java编译器将模板作为代码来处理。 (最后但并非最不重要的一点是,因为使用占位符的模板可能不是语法上正确的Java代码片段。)我们需要一个新的/*行,我们不想将其插入模板。
因此,此行之前包含// SKIP的行。 (跳过行在命令前可以具有// 。)

您可以在生成的代码中看到的结果。 所有方法都具有正确的JavaDoc文档。

SEP1SEP2

遍历值已与占位符的名字分开,并且| 值列表。 例如,上面的示例包含

 LOOP type ,variable=int,i|long,l|short,s 

两个占位符名称typevariable ,每个占位符三个值。
占位符不需要包含特殊字符,如果它们是标准标识符,则最好。 但是,这些值可能包含逗号或竖线。 在这种情况下,你可以重新定义字符串(不仅是单个字符),模板LOOP命令可以改用单一字符串,|

例如线

 SEP1 / 

表示名称和值应以/分隔,而不是一个和

 SEP2 & 

的值的列表应当由一个字符分隔&
串。 SEP1SEP2仅在它们之前
LOOP命令,它们仅对所使用的模板有效。遵循上述命令, LOOP示例将如下所示:

 LOOP type /variable =int /i &long /l &short /s 

这样,没有什么可以阻止我们添加另一个值列表

 LOOP type /variable =int /i &long /l &short /s &byte,int /z 

最终将导致示例模板出现语法错误,但演示了重新定义名称和值列表分隔符的要点。

组态

生成器由Geci框架支持的配置工具实现,并且所有参数都是可配置的。 您可以在类的注释中或在编辑器折叠参数中,重新定义与模板的开始,结束,跳过等行相匹配的正则表达式,在创建生成器对象的单元测试中。

带走

迭代生成器是一种非常易于使用的生成器,用于创建重复的代码。 这也是主要的危险:您应该足够强大才能找到更好的解决方案,并仅在最佳解决方案时才使用它。

翻译自: https://www.javacodegeeks.com/2019/11/repeated-code.html

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐