当前位置: 首页 > 面试题库 >

为什么打印java对象得到SomeType@2f92e0f4这样的结果?

酆乐湛
2023-03-14
问题内容

为什么打印java对象得到类似SomeType@2f92e0f4这样的结果?


问题答案:

背景

所有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 -类的名称,即类MyType在package中com.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;
  }
}

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

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

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

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

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

自动生成输出

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

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

  • ToStringBuilder来自Apache Commons Lang
  • MoreObjects.ToStringHelper来自Google Guava
  • @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()如上所述的尼斯。



 类似资料:
  • 问题内容: 我生成了x的两个矩阵: 第一矩阵:和。 第二矩阵:和。 使用以下代码,第一个矩阵花费了8.52秒完成: 使用此代码,第二个矩阵花费了259.152秒来完成: 运行时间显着不同的原因是什么? 正如评论所说,仅打印需要秒,而给。 正如其他指出它对他们正常工作的人一样,例如,我尝试了Ideone.com,这两段代码以相同的速度执行。 测试条件: 我从 Netbeans 7.2 运行了此测试,

  • 问题内容: 我有以下代码行: 我希望看到的结果是0.555,但是它将打印出零。有人可以帮我理解为什么会这样吗?我目前正在学习编程,并感谢您的帮助。 谢谢! 问题答案: 发生这种情况是因为您在不知不觉中正在执行Integer Division。为了使计算更快,计算机在不涉及十进制数的情况下使用整数除法,因此会丢失十进制值。 试试看: 要么 要么 要么 要么

  • 我想知道当您试图捕获StackOverflowError时会发生什么,并提出了以下方法: 现在我的问题是: 为什么这个方法打印'4'? 我想可能是因为在调用堆栈上需要3个段,但我不知道3这个数字来自哪里。当您查看的源代码(和字节码)时,通常会导致比3个多得多的方法调用(因此调用堆栈中的3个段是不够的)。如果这是因为Hotspot VM应用的优化(方法内联),我想知道在另一个VM上结果是否会有所不同

  • 我试图使用流按国籍对我的对象进行分组并打印出来。 但是它说:"不能解析方法'println"

  • 上述代码是我在进行 ramda 函数库的 pipe 实现过程中的代码。 我无法通过测试用例 performs left-to-right function composition 和 can be applied to one argument 我发现是因为我是实现的代码 f的长度都是0, 所以我想问什么原因造成的这个问题,如何解决? ramda - test -pipe