我有一个类定义如下:

public class Person {
  private String name;

  // constructor and getter/setter omitted
}

我试图打印类的一个实例:

System.out.println(myPerson);

但我得到了以下输出:com.foo.Person@2f92e0f4。

当我试图打印Person对象数组时,也发生了类似的事情:

Person[] people = //...
System.out.println(people); 

我得到了输出:[Lcom.foo.Person;@28a418fc

这个输出意味着什么?如何更改此输出,使其包含我的人的名字?我如何打印对象的集合?

注意:这是一个关于这个主题的规范问答。


当前回答

对于“deep”toString(),有一个基于JSON的答案(Jackson, GSON等)的替代方案:来自Apache Commons Lang 3库的ReflectionToStringBuilder,使用RecursiveToStringStyle或MultilineRecursiveToStringStyle。代码示例:

System.out.println("My object: " +
    ReflectionToStringBuilder.toString(theObject, new RecursiveToStringStyle()));

输出示例:

// RecursiveToStringStyle
Person@7f54[name=Stephen,age=29,smoker=false,job=Job@43cd2[title=Manager]]

// MultilineRecursiveToStringStyle
Person@7f54[
  name=Stephen,
  age=29,
  smoker=false,
  job=Job@43cd2[
    title=Manager
  ]
]

其他回答

在类上使用Lombok @Data注释将提供getter、setter、toString和hashcode。使用Lombok更好,因为它处理样板代码。

在intellij中,您可以通过按alt+inset然后选择toString()自动生成toString方法,这里是一个测试类的输出:

public class test  {
int a;
char b;
String c;
Test2 test2;

@Override
public String toString() {
    return "test{" +
            "a=" + a +
            ", b=" + b +
            ", c='" + c + '\'' +
            ", test2=" + test2 +
            '}';
 }
}

正如您所看到的,它通过连接类的几个属性来生成一个String,对于原语,它将打印它们的值,对于引用类型,它将使用它们的类类型(在本例中是Test2的to String方法)。

默认情况下,Java中的每个类都有toString()方法,如果将该类的某个对象传递给System.out.println(),则会调用该方法。默认情况下,该调用返回该对象的className@hashcode。

{
    SomeClass sc = new SomeClass();
    // Class @ followed by hashcode of object in Hexadecimal
    System.out.println(sc);
}

您可以重写类的toString方法以获得不同的输出。请看这个例子

class A {
    String s = "I am just a object";
    @Override
    public String toString()
    {
        return s;
    }
}

class B {
    public static void main(String args[])
    {
        A obj = new A();
        System.out.println(obj);
    }
}

如果您正在使用项目Lombok,您可以使用@ToString注释并生成一个标准的toString()方法,而无需添加样板。

import lombok.ToString;

@ToString
public class LoginDto {
  private String user;
  private String pass;
}
...
System.out.println(loginDto.toString());
// LoginDto(user=x@xxx.x, pass=xxxxx)

我更喜欢使用一个实用函数,它使用GSON将Java对象反序列化为JSON字符串。

/**
 * This class provides basic/common functionalities to be applied on Java Objects.
 */
public final class ObjectUtils {

    private static final Gson GSON = new GsonBuilder().setPrettyPrinting().create();

    private ObjectUtils() {
         throw new UnsupportedOperationException("Instantiation of this class is not permitted in case you are using reflection.");
    }

    /**
     * This method is responsible for de-serializing the Java Object into Json String.
     *
     * @param object Object to be de-serialized.
     * @return String
     */
    public static String deserializeObjectToString(final Object object) {
        return GSON.toJson(object);
    }
}