
在Java编程中,我们经常需要将对象的状态以字符串形式输出,例如在控制台打印、日志记录或调试时。然而,当直接对包含自定义对象的数组调用 `Arrays.toString()` 方法时,我们可能会发现输出的并非对象内部的实际数据,而是一串形如 `ClassName@hashCode` 的字符串,即对象的哈希码。这通常是由于对 `toString()` 方法的默认行为理解不足所导致的。
理解Java对象的默认字符串表示
在java中,所有类都隐式或显式地继承自 java.lang.Object 类。object 类提供了一个默认的 tostring() 方法,其实现通常是返回对象的运行时类名 @ 符号,后跟对象的哈希码的无符号十六进制表示。例如:com.example.student@1a2b3c4d。
当您创建一个自定义类(如 Student 类)并将其对象放入数组中,然后尝试使用 Arrays.toString() 打印该数组时,Arrays.toString() 会遍历数组中的每个元素,并调用每个元素的 toString() 方法来获取其字符串表示。如果您的自定义类没有重写 toString() 方法,那么就会调用 Object 类的默认实现,从而打印出对象的哈希码,而不是您期望的字段内容。
解决方案:重写 toString() 方法
要解决这一问题,核心在于在自定义类中重写 toString() 方法。通过重写,我们可以自定义对象在被转换为字符串时应该如何呈现其内部状态。
在 Student 类中,我们可以这样重写 toString() 方法:
@Override public String toString() { return "id: " + this.id + ", fName: " + this.fName + ", lName: " + this.lName + ", score: " + this.score; }
这段代码的作用是:
立即学习“Java免费学习笔记(深入)”;
- @Override 注解:这是一个可选的注解,用于告诉编译器该方法是重写父类(这里是 Object 类)的方法。如果方法签名不匹配,编译器会报错,有助于防止拼写错误或签名不一致的问题。
- 方法体:我们在这里构建了一个包含 id、fName、lName 和 score 字段值的字符串。通过字符串拼接,我们可以清晰地展示 Student 对象的各个属性。
完整示例代码
下面是包含 Student 类及其重写 toString() 方法的完整示例代码。此代码还包含了学生数据的输入、验证和按分数排序的功能,展示了如何在实际应用中集成 toString() 方法。
import java.util.Arrays; import java.util.Comparator; import java.util.Scanner; public class Main { // 学生类定义 private static class Student { String fName; // 名 String lName; // 姓 int id; // 学生ID int score; // 分数 public Student(String fName, String lName, int id, int score) { this.fName = fName; this.lName = lName; this.id = id; this.score = score; } public int getScore() { return score; } // 重写toString()方法,使其打印对象内容而非哈希码 @Override public String toString() { return "id: " + this.id + ", fName: " + this.fName + ", lName: " + this.lName + ", score: " + this.score; } } // 检查字符串是否全为字母 public static boolean isAlphabetic(String str) { char[] charArray = str.toCharArray(); for (char c : charArray) { if (!Character.isLetter(c)) { return false; } } return true; } public static void main(String[] args) { Scanner input = new Scanner(System.in); System.out.println("请输入学生数量:"); int k = input.nextInt(); // 学生数量 Student[] students = new Student[k]; for (int i = 0; i < k; ) { System.out.println("请输入学生 " + (i + 1) + " 的ID:"); int id = input.nextInt(); System.out.println("请输入学生 " + (i + 1) + " 的名:"); String fName = input.next(); while (!isAlphabetic(fName)) { System.out.println("输入错误!名必须是字母。请重新输入:"); fName = input.next(); } System.out.println("请输入学生 " + (i + 1) + " 的姓:"); String lName = input.next(); while (!isAlphabetic(lName)) { System.out.println("输入错误!姓必须是字母。请重新输入:"); lName = input.next(); } System.out.println("请输入学生 " + (i + 1) + " 的分数:"); int score = input.nextInt(); students[i++] = new Student(fName, lName, id, score); } // 定义按分数排序的比较器 class ByScoreComparator implements Comparator<Student> { @Override public int compare(Student s1, Student s2) { // 降序排序,所以s2的分数在前 return Integer.compare(s2.getScore(), s1.getScore()); } } // 使用自定义比较器对学生数组进行排序 Arrays.sort(students, new ByScoreComparator()); System.out.println("n排序后的学生信息:"); // 打印数组,此时会调用Student类的toString()方法 System.out.println(Arrays.toString(students)); input.close(); } }
注意事项:
- 在上述 ByScoreComparator 中,为了实现降序排序,我们将 s2.getScore() 放在了 s1.getScore() 之前。如果需要升序,则应为 Integer.compare(s1.getScore(), s2.getScore())。
- Scanner 资源在使用完毕后应关闭,以避免资源泄露。在 main 方法的末尾添加 input.close(); 是一个好习惯。
运行结果与分析
假设输入以下学生数据:
- 学生1: ID=1, 名=Mo, 姓=Nee, 分数=900
- 学生2: ID=2, 名=Lee, 姓=Jen, 分数=600
在运行上述代码并输入数据后,预期的输出将是:
排序后的学生信息: [id: 1, fName: Mo, lName: Nee, score: 900, id: 2, fName: Lee, lName: Jen, score: 600]
可以看到,Arrays.toString(students) 现在输出了每个 Student 对象的详细内容,而不是其哈希码。这是因为 Student 类中重写的 toString() 方法被成功调用。
总结与最佳实践
重写 toString() 方法是Java中一个非常重要的实践,它不仅解决了打印对象哈希码的问题,还带来了以下好处:
- 调试便利性: 在调试过程中,通过打印对象可以快速了解其当前状态。
- 日志记录: 在应用程序日志中输出对象信息时,重写的 toString() 方法提供了有意义的上下文。
- 可读性: 使代码输出更具可读性和可理解性,尤其是在处理复杂对象时。
建议:
- 始终在自定义类中重写 toString() 方法,尤其是在这些对象可能被打印、记录或用于调试的场景。
- 在 toString() 方法中包含所有关键字段,以便清晰地表示对象的状态。
- 保持 toString() 方法的输出简洁明了,但信息量足够。
- 使用 StringBuilder 或 StringJoiner(Java 8+)来构建复杂的字符串,以提高性能和可读性,尤其是在涉及大量字段或循环时。
通过遵循这些实践,您可以确保Java对象在被转换为字符串时始终提供有意义且易于理解的信息。


