我有一个类定义如下:

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
  ]
]

其他回答

对于“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
  ]
]

我在Spring 5中使用Jackson做到了这一点。根据对象的不同,它可能不是在所有情况下都有效。

import com.fasterxml.jackson.databind.ObjectMapper;
    
ObjectMapper mapper = new ObjectMapper();
System.out.println(mapper.writeValueAsString(yourObject));

输出是这样的

{
  "id" : 1,
  "fieldOne" : "string"
}

这里有更多使用Jackson的例子

如果你用GSON代替,它可能看起来像

Gson gson = new Gson();
System.out.println(gson.toJson(yourObject));

我更喜欢使用一个实用函数,它使用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);
    }
}

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

默认情况下,Java中的每个对象都有toString()方法,该方法输出ObjectType@HashCode。

如果您想要更多有意义的信息,那么您需要重写类中的toString()方法。

public class Person {
  private String name;

  // constructor and getter/setter omitted

  // overridding toString() to print name
  public String toString(){
     return name;  
  }
}

现在当你打印person对象时使用System.out.prtinln(personObj);它将打印人名,而不是类名和hashcode。

在第二种情况下,当你试图打印数组时,它会打印数组类型[Lcom.foo.Person;


如果要打印人名,有很多方法。

您可以编写自己的函数,迭代每个人并输出

void printPersonArray(Person[] persons){
    for(Person person: persons){
        System.out.println(person);
    }
}

你可以使用Arrays.toString()来打印它。这对我来说似乎是最简单的。

 System.out.println(Arrays.toString(persons));
 System.out.println(Arrays.deepToString(persons));  // for nested arrays  

你可以用java 8的方式打印它(使用流和方法引用)。

 Arrays.stream(persons).forEach(System.out::println);

也许还有其他方法。希望这能有所帮助。:)