我正在尝试使用com.sun.codemodel.JCodeModel生成Java对象。
我已经成功地生成了hashcode()和equals()方法,但是我正在与toString()做斗争;
我需要以下toString()实现
return "ClassName [field1 = " + field1 + ", field2 = " + field2 ... ", fieldN = " + fieldN + "]";如何创建包含与JCodeModel连接的JExpr.lit(field1.name())的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);发布于 2017-12-18 18:10:36
这里的关键点可能是,您可以使用JExpression方法将多个+对象与+运算符组合起来。
下面是一个包含简单示例类定义的示例,以及一个泛化生成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]这应该是你所期望的,根据你的例子。
https://stackoverflow.com/questions/47872206
复制相似问题