java中对象为null时的打印输出方式
作者:衣有尘
java对象为null时的打印输出
在写代码时,突然好奇当一个对象为空时,打印输出为何是null,便作此小记。
public static void main(String[] args) {
String a = null;
String b = "null";
Object c = null;
System.out.println(a);
System.out.println(b);
System.out.println(c);
}我们现在有a\b\c三个对象,以上代码的输出如下,均为null。
null
null
null
为了了解为何打印为null,需要查看println()方法的源码
public void println(String x) {
synchronized (this) {
print(x);
newLine();
}
}
public void print(String s) {
write(String.valueOf(s));
}
public static String valueOf(Object obj) {
return (obj == null) ? "null" : obj.toString();
}可以看到println()方法实际上调用了两个方法。
而print方法调用了String.valueOf(s)这个静态方法,该方法在将对象转换为String时,会判断对象是否为空(即为null),若对象为空,则直接返回字符串"null"。
null对象打印为什么是null?
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();
}
}总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。
