我正在尝试使用com.sun.codemodel.JCodeModel
生成Java Value Objects。
我已设法生成hashcode()
和equals()
方法,但我正在努力toString()
;
我需要以下toString()
实施
return "ClassName [field1 = " + field1 + ", field2 = " + field2 ... ", fieldN = " + fieldN + "]";
如何创建包含JCodeModel
与JExpression
连接的JExpr.lit(field1.name())
JExpr.ref(fieldVar.name())
?
所有我设法做的就是生成一个字符串文字,类似于: -
return "ClassName [field1 = field1 + field2 = field2 ... fieldN = + fieldN + ]";
到目前为止,我的骨架方法是什么: -
final Map<String, JFieldVar> fields = jclass.fields();
final JMethod toString = jclass.method(JMod.PUBLIC, String.class, "toString");
final Set<String> excludes = new HashSet<String>(Arrays.asList(ruleFactory.getGenerationConfig().getToStringExcludes()));
final JBlock body = toString.body();
for (JFieldVar fieldVar : fields.values()) {
if (excludes.contains(fieldVar.name()) || ((fieldVar.mods().getValue() & JMod.STATIC) == JMod.STATIC)) {
continue;
}
??????????????
}
body._return(?????????);
toString.annotate(Override.class);
答案 0 :(得分:1)
此处的关键点可能是您可以使用JExpression
方法将多个+
对象与JExpression#plus
运算符组合在一起。
这是一个包含简单示例类定义的示例,以及一般生成toString
方法的方法:
import java.util.Arrays;
import java.util.Collection;
import java.util.Map;
import com.sun.codemodel.CodeWriter;
import com.sun.codemodel.JBlock;
import com.sun.codemodel.JCodeModel;
import com.sun.codemodel.JDefinedClass;
import com.sun.codemodel.JExpr;
import com.sun.codemodel.JExpression;
import com.sun.codemodel.JFieldVar;
import com.sun.codemodel.JMethod;
import com.sun.codemodel.JMod;
import com.sun.codemodel.writer.SingleStreamCodeWriter;
public class CodeModelToStringTest
{
public static void main(String[] args) throws Exception
{
JCodeModel codeModel = new JCodeModel();
JDefinedClass definedClass = codeModel._class("com.example.Example");
definedClass.field(JMod.PUBLIC, String.class, "exampleString");
definedClass.field(JMod.PROTECTED, int.class, "exampleInt");
definedClass.field(JMod.PRIVATE, float.class, "exampleFloat");
definedClass.field(JMod.PRIVATE, String.class, "excludedString");
definedClass.field(JMod.STATIC, String.class, "staticString");
createToStringMethod(definedClass, Arrays.asList("excludedString"));
CodeWriter codeWriter = new SingleStreamCodeWriter(System.out);
codeModel.build(codeWriter);
}
private static void createToStringMethod(
JDefinedClass definedClass,
Collection<String> excludedFieldNames)
{
Map<String, JFieldVar> fields = definedClass.fields();
JMethod toString =
definedClass.method(JMod.PUBLIC, String.class, "toString");
toString.annotate(Override.class);
JBlock body = toString.body();
JExpression expression = JExpr.lit(definedClass.name() + " [");
boolean first = true;
for (JFieldVar fieldVar : fields.values())
{
if ((fieldVar.mods().getValue() & JMod.STATIC) == JMod.STATIC)
{
continue;
}
if (excludedFieldNames.contains(fieldVar.name()))
{
continue;
}
if (!first)
{
expression = expression.plus(JExpr.lit(", "));
}
expression = expression.plus(JExpr.lit(fieldVar.name()+" = "));
expression = expression.plus(JExpr.ref(fieldVar.name()));
first = false;
}
expression = expression.plus(JExpr.lit("]"));
body._return(expression);
}
}
此处显示了使用toString
方法生成的类:
package com.example;
public class Example {
public String exampleString;
protected int exampleInt;
private float exampleFloat;
private String excludedString;
static String staticString;
@Override
public String toString() {
return ((((((((("Example ["+"exampleString = ")+ exampleString)+", ")+"exampleInt = ")+ exampleInt)+", ")+"exampleFloat = ")+ exampleFloat)+"]");
}
}
CodeModel在每个二进制操作周围插入(
括号)
这一事实导致代码看起来不那么漂亮。但这是可以理解的:否则,他们必须考虑运算符优先级,并且使用和代码生成本身可能远更难。
但是,此toString
方法的结果将是
Example [exampleString = null, exampleInt = 0, exampleFloat = 0.0]
根据您的示例,应该是您所期望的。