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(); } }
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。