为什么 ToStringBuilder 工作不一致?
Why does ToStringBuilder work inconsistently?
在下面的代码中,为什么包含 System.out.println(person);
的两行会产生不同的输出?第二行间接调用方法 Job.toString
产生字符串 "Manager"
,但第一行神秘地没有产生 Job@28f67ac7
。 person.put("a", "b");
之间的线在我看来应该没有任何区别。
代码:
import java.util.*;
import org.apache.commons.lang3.builder.*;
class Job extends HashMap<String, String> {
@Override public String toString() {
return "Manager";
}
}
class Person extends HashMap<String, String> {
Job job;
Person() {
this.job = new Job();
}
@Override public String toString() {
return ToStringBuilder.reflectionToString(this);
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println(person);
person.put("a", "b");
System.out.println(person);
}
}
控制台:
Person@2b80d80f[job=Job@28f67ac7,threshold=0,loadFactor=0.75]
Person@2b80d80f[job=Manager,threshold=12,loadFactor=0.75]
2 个因素导致输出变化:
- ToStringBuilder 避免在相等实例上调用 toString() 以避免无限递归。
- 你的 Person 和 Job 类 继承了 HashMap 的 equals() 方法,导致
new Person().equals(new Job()) == true
这意味着当您的示例中的 person 和 person.job 保持彼此相等时,ToStringBuilder 将不会调用 person.job.toString(),但是当映射内容更改时,person.job .toString() 将被调用。
在下面的代码中,为什么包含 System.out.println(person);
的两行会产生不同的输出?第二行间接调用方法 Job.toString
产生字符串 "Manager"
,但第一行神秘地没有产生 Job@28f67ac7
。 person.put("a", "b");
之间的线在我看来应该没有任何区别。
代码:
import java.util.*;
import org.apache.commons.lang3.builder.*;
class Job extends HashMap<String, String> {
@Override public String toString() {
return "Manager";
}
}
class Person extends HashMap<String, String> {
Job job;
Person() {
this.job = new Job();
}
@Override public String toString() {
return ToStringBuilder.reflectionToString(this);
}
}
class Test {
public static void main(String[] args) {
Person person = new Person();
System.out.println(person);
person.put("a", "b");
System.out.println(person);
}
}
控制台:
Person@2b80d80f[job=Job@28f67ac7,threshold=0,loadFactor=0.75]
Person@2b80d80f[job=Manager,threshold=12,loadFactor=0.75]
2 个因素导致输出变化:
- ToStringBuilder 避免在相等实例上调用 toString() 以避免无限递归。
- 你的 Person 和 Job 类 继承了 HashMap 的 equals() 方法,导致
new Person().equals(new Job()) == true
这意味着当您的示例中的 person 和 person.job 保持彼此相等时,ToStringBuilder 将不会调用 person.job.toString(),但是当映射内容更改时,person.job .toString() 将被调用。