如何在不显示“SomeType @ 2f92e0f4”的情况下打印我的Java对象?

我有一个类定义如下:

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

这个输出是什么意思? 我如何改变这个输出,使其包含我的人的名字? 我如何打印我的对象的集合?

注意 :这是针对此主题的规范问答。


背景

所有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 - 类的名称,即类com.foo MyType
  • @ - 将字符串连接在一起
  • 2f92e0f4对象的哈希码。
  • 数组类的名称看起来有点不同,这在Class.getName()的Javadocs中有很好的解释。 例如, [Ljava.lang.String意思是:

  • [ - 一维数组(与[[[[[ etc.]
  • 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;
      }
    }
    

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

    请记住, toString()只是将对象转换为字符串的一种方式。 通常,这个输出应该以清晰和简洁的方式充分描述你的对象。 对于我们的Person类,更好的toString()可能是:

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

    哪个会打印,例如Person[name=Henry] 。 这是一个非常有用的调试/测试数据。

    如果您只想关注对象的一个​​方面或包含大量爵士格式,则最好定义一个单独的方法,例如String toElegantReport() {...}


    自动生成输出

    许多IDE支持基于类中的字段自动生成toString()方法。 例如,请参阅Eclipse和IntelliJ的文档。

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

  • 来自Apache Commons Lang的ToStringBuilder

  • 来自Google Guava的MoreObjects.ToStringHelper

  • 来自Project Lombok的@ToString注释


  • 打印对象组

    所以你为你的类创建了一个很好的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]
    

    所以你只需要确保你的列表元素如上所述定义了一个很好的toString()


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

    ReflectionToStringBuilder.toString(object)
    

    java中的每个类都有默认的toString()方法,如果你传递一个类的某个对象给System.out.println()调用它。 当您尝试打印某个类的对象时, System.out.println()方法将调用该类的toString() ,该类将返回该对象的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);
        }
    }
    
    链接地址: http://www.djcxy.com/p/36901.html

    上一篇: How do I print my Java object without getting "SomeType@2f92e0f4"?

    下一篇: How to dispatch a Redux action with a timeout?