先看这样一个例子:
public class Demo {
static int i = 1;
public static void main(String[] args) {
System.out.println("love" + new Demo());
Demo a = new Demo();
a.i++;
System.out.println("me " + a.i);
}
@Override
public String toString(){
System.out.println("I ");
return " Java ";
}
}
原先认为输出结果是:
love I java
me 2
实际输出结果为:
I
love java
me 2
相当有意思!
这里引出两个问题
System.out.println()的真正含义
- 为什么println(new Obj());会执行toString方法
先了解下第一个问题:System.out.println()的真正含义
我们常常用 System.out.println(); 来输出字符串,也许我们都已经猜到println()是方法名,但System是什么,out又是什么呢?
经过查阅资料后了解到:
System是java.lang里面的一个类;
而out就是System里面的一个静态数据成员,而且这个成员是java.io.PrintStream类的引用。如下图,被关键字static 修饰的成员可以直接通过"类名.成员名"来引用,而无需创建类的实例。所以System.out是调用了System类的静态数 据成员out。
println()就是java.io.PrintStream类里的一个方法,它的作用是向控制台输出信息。因为System.out是 java.io.PrintStream类的实例的引用,所以可以通过 System.out.println(); 来调用此方法。
第二个问题:在调用System.out.println()打印一个新new的对象时,为什么会调用toString方法?
当使用System.out.println()里面为一个对象的引用时,自动调用toString方法将对象打印出来。如果重写了tostring方法则调用重写的toString 方法。
//因为System.out.println()的这个方法源码中调用了String.valueOf(Objec o),
public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}
//而String.valueOf(x)的源码就是去调用该对象的toString()方法,源码如下:
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}
//Object 类的toString()源码
public String toString()
返回该对象的字符串表示。通常,toString 方法会返回一个“以文本方式表示”此对象的字符串。结果应是一个简明但易于读懂的信息表达式。建议所有子类都重写此方法。
Object 类的 toString 方法返回一个字符串,该字符串由类名(对象是该类的一个实例)、at 标记符“@”和此对象哈希码的无符号十六进制表示组成。换句话说,该方法返回一个字符串,它的值等于:
getClass().getName() + '@' + Integer.toHexString(hashCode())
关于数据存放在内存中的哪里:Demo a;(a 指向的对象) 和i 存放在栈中,而堆中存放 new Demo();
这里引入一个知识点:无论是类方法还是静态方法,都是在栈区调用和执行的。
所以在执行 System.out.println("love" + new Demo());时,先new Demo()执行Demo重写的toString()方法,就会有了输出 "I " " love" " Java"的结果。