如何在不获得“SomeType@2f92e0f4”的情况下打印我的 Java 对象?背景自定义输出自动生成输出打印对象组

2022-08-31 04:42:09

我有一个定义如下的类:

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

此输出是什么意思?如何更改此输出以使其包含我的人的姓名?如何打印对象的集合?

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


答案 1

背景

所有 Java 对象都有一个方法,当您尝试打印该对象时,将调用该方法。toString()

System.out.println(myObject);  // invokes myObject.toString()

此方法在 Object 类(所有 Java 对象的超类)中定义。Object.toString() 方法返回一个外观相当丑陋的字符串,该字符串由类名、符号和十六进制对象的哈希码组成。其代码如下所示:@

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

因此,这样的结果可以解释为:com.foo.MyType@2f92e0f4

  • com.foo.MyType- 类的名称,即类在包中。MyTypecom.foo
  • @- 将字符串连接在一起
  • 2f92e0f4对象的哈希代码。

数组类的名称看起来有点不同,这在 Javadocs for Class.getName() 中得到了很好的解释。例如,表示:[Ljava.lang.String

  • [- 一维数组(相对于或等)[[[[[
  • L- 数组包含类或接口
  • java.lang.String- 数组中对象的类型

自定义输出

若要在调用 时打印不同的东西,必须在自己的类中重写该方法。下面是一个简单的示例:System.out.println(myObject)toString()

public class Person {

  private String name;
  
  // constructors and other methods omitted
  
  @Override
  public String toString() {
    return name;
  }
}

现在,如果我们打印一个 ,我们会看到它们的名字而不是 。Personcom.foo.Person@12345678

请记住,这只是将对象转换为字符串的一种方法。通常,此输出应以清晰简洁的方式完整描述您的对象。对于我们的班级来说,更好的可能是:toString()toString()Person

@Override
public String toString() {
  return getClass().getSimpleName() + "[name=" + name + "]";
}

这将打印,例如, .这是调试/测试中非常有用的数据。Person[name=Henry]

如果你想只关注对象的一个方面,或者包括很多爵士乐的格式,你最好定义一个单独的方法,例如。String toElegantReport() {...}


自动生成输出

许多 IDE 都支持根据类中的字段自动生成方法。例如,请参阅 EclipseIntelliJ 的文档。toString()

几个流行的Java库也提供了此功能。一些示例包括:


打印对象组

所以你已经为你的班级创造了一个很好的。如果将该类放入数组或集合中,会发生什么情况?toString()

阵 列

如果您有一个对象数组,则可以调用 Arrays.toString() 来生成数组内容的简单表示形式。例如,考虑以下对象数组:Person

Person[] people = { new Person("Fred"), new Person("Mike") };
System.out.println(Arrays.toString(people));

// Prints: [Fred, Mike]

注意:这是对 Arrays 类中调用的静态方法的调用,这与我们上面讨论的内容不同。toString()

如果您有一个多维数组,则可以使用 Arrays.deepToString() 来实现相同类型的输出。

收集

大多数集合将基于对每个元素的调用生成漂亮的输出。.toString()

List<Person> people = new ArrayList<>();
people.add(new Person("Alice"));
people.add(new Person("Bob"));    
System.out.println(people);

// Prints [Alice, Bob]

因此,您只需要确保您的列表元素定义了上面讨论的 nice。toString()


答案 2

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

ReflectionToStringBuilder.toString(object)