JAVA中有八种基本数据类型,他们在初始化的时候会被赋予初始值。
byte -> 0
short -> 0
int -> 0
long -> 0
char -> " "(看起来像是一个空格)
float -> 0.0
double -> 0.0
boolean -> false
引用类型对象初始化会被赋予初始值:null
String s = null;
try{
s.toString();
} catch(NullPointerException e){
e.printStackTrace();
}
System.out.println(s);
执行结果如下:
java.lang.NullPointerException
at GjcTest.main(GjcTest.java:5)
null
问题1:String类型null对象为什么会被打印成null?
当我们调用null对象的方法时,会抛出异常。但是我们打印null对象却不会抛出异常,感觉是println方法对null对象做了什么处理。
println方法源码如下:
public void println(String x) {
synchronized (this) {
print(x);
newLine();
}
}
嗯,答案可能在方法print中,我们打开print方法源码:
public void print(String s) {
if (s == null) {
s = "null";
}
write(s);
}
嗯,简单粗暴的处理方式,如果传入String对象是null,则直接打印字符串“null”。
那如果是引用类型呢?我们就以Integer类作为示例。
Integer i = null;
try{
i.toString();
} catch(NullPointerException e){
e.printStackTrace();
}
System.out.println(i);
执行结果如下:
java.lang.NullPointerException
at GjcTest.main(GjcTest.java:5)
null
问题2:非String类型null对象为什么会被打印成null?
我们先看一下println源码:
public void println(Object x) {
String s = String.valueOf(x);
synchronized (this) {
print(s);
newLine();
}
}
第一感觉是print方法应该是对null对象进行了处理的,我们直接打开print方法源码:
public void print(String s) {
if (s == null) {
s = "null";
}
write(s);
}
嗯?怎么调用的是参数类型是String类型的方法?哦,原来在println方法中,调用print方法参数类型是String类的,那么问题应该是在println源码中第2行valueOf方法中。valueOf方法源码如下:
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}
这个问题就解决了,如果传入的引用类型对象是null,那么返回字符串“null”,否则调用对象的toString方法。
总结
String类型null对象处理方式:判断对象是否是null,如果是null,打印字符串“null”;如果不是null,打印字符串。
非String类型null对象处理方式:判断对象是否是null,如果是null,打印字符串“null”;如果不是null,调用对象的toString方法,并打印字符串。
我的疑问
如图所示:println(Object o)处理逻辑是调用String.valueOf()方法,判断对象o是否为空并返回需要打印的字符串,然后调用print(String s)打印出来。为什么println(Object o)不直接调用print(Object o)呢,代码如下。是设计者就是这样设计的,还是有什么原因?
public void println(Object x) {
synchronized (this) {
print(x);
newLine();
}
}