我有一个类定义如下:

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更好,因为它处理样板代码。

在Eclipse中, 去上课吧, 右击->源->生成toString();

它将重写toString()方法并打印该类的对象。

我认为apache提供了一个更好的util类,它提供了一个函数来获取字符串

ReflectionToStringBuilder.toString(object)

如果您正在使用项目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)

如果你看一下Object类(Java中所有类的父类),toString()方法实现是

    public String toString() {
       return getClass().getName() + "@" + Integer.toHexString(hashCode());
    }

当你在Java中打印任何对象时,toString()将被调用。现在由你决定,如果你重写toString(),那么你的方法将调用其他Object类的方法调用。