由一个简单的String c=a+b的Java问题引发一点想法

简介:

    今天在我的一个qq群里有人问了这样一个问题。

 
  1. private static String a; 
  2. private static String b; 
  3.  
  4.   public static void main(String[] args) { 
  5.        String c = a+b; 
  6.        System.out.println(c); 
  7.     } 

输出是”nullnull”问为什么是这样。

其实问题并不复杂,很多同学也觉得自己知道原因,遂不予关注。但是我相信还是有初学Java的同学在这里是存在误会的。很典型的误会就是如一个群里的朋友说的String类型的变量如果没有显示初始化,默认的值就是”null”。支持的理由就是

 
  1. private static String a; 
  2.     public static void main(String[] args) { 
  3.        System.out.println(a); 
  4.     } 

输出是“null”。

这个现实确实很容易迷惑一些初学的人,包括我也忽略了挺久。其实呢证明这种想法错误很简单。如果默认值是“null”,那么就意味着,该变量不是空(null)。而是字符串的”null”

 
  1. private static String a; 
  2.     public static void main(String[] args) { 
  3.        System.out.println(a==null); 
  4.        System.out.println("null".equals(a)); 
  5.     } 

上述代码输出分别是true,false

说明String类型变量a,其实是空(null),而并没有被赋值。那么打印出null是为什么呢?我们查看PrintStream的源码就很清晰的明白了,其实是Javaprintln的时候进行了处理。

 
  1. public void print(String s) { 
  2.     if (s == null) { 
  3.         s = "null"
  4.     } 
  5.     write(s); 
  6.     } 

回到开头的问题,既然没有初始化赋值,那么输出为什么是”nullnull”,两个”null”连接的结果呢。这里略微细说一下,查看编译过的class文件,我们可以看到

 
  1. public static void main(java.lang.String[] args); 
  2.      0  new java.lang.StringBuilder [19
  3.      3  dup 
  4.      4  getstatic cn.home.pratice.jdk.string.StringMain.a : java.lang.String [21
  5.      7  invokestatic java.lang.String.valueOf(java.lang.Object) : java.lang.String [23
  6.     10  invokespecial java.lang.StringBuilder(java.lang.String) [29
  7.     13  getstatic cn.home.pratice.jdk.string.StringMain.b : java.lang.String [32
  8.     16  invokevirtual java.lang.StringBuilder.append(java.lang.String) : java.lang.StringBuilder [34
  9.     19  invokevirtual java.lang.StringBuilder.toString() : java.lang.String [38
  10.     22  astore_1 [c] 
  11.     23  getstatic java.lang.System.out : java.io.PrintStream [42
  12.     26  aload_1 [c] 
  13.     27  invokevirtual java.io.PrintStream.println(java.lang.String) : void [48
  14.     30  return       System.out.println(c); 
  15.     } 

 String的相加实际在变异后被处理成了StringBuilderappend.(注:我的JDK1.6.0_u29)。那么好,我们就应该查看StringBuilder的源码,发现是调用的父类里的方法,继续查看,道理就在这里。

 
  1. public StringBuilder append(String str) { 
  2.     super.append(str); 
  3.         return this
  4.  
  5. public AbstractStringBuilder append(String str) { 
  6.     if (str == null) str = "null"
  7.         int len = str.length(); 
  8.     if (len == 0return this
  9.     int newCount = count + len; 
  10.     if (newCount > value.length) 
  11.         expandCapacity(newCount); 
  12.     str.getChars(0, len, value, count); 
  13.     count = newCount; 
  14.     return this
  15.     } 

原来也是对空 null ,进行了特殊的处理,那么输出是 ”nullnull” ,自然也就明白了。

    这里我想说的是,很多问题,可能表面上很简单,或者我们可能会有很多想当然的想法,不过还是眼见为实,而且所有代码都放在那里,我们为什么不勤快的多翻开看看其中的实现,道理自然就在眼前。多动手,丰衣足食:)




     本文转自mushiqianmeng 51CTO博客,原文链接:http://blog.51cto.com/mushiqianmeng/723175,如需转载请自行联系原作者



相关文章
|
16天前
|
Java API 索引
Java基础—笔记—String篇
本文介绍了Java中的`String`类、包的管理和API文档的使用。包用于分类管理Java程序,同包下类无需导包,不同包需导入。使用API时,可按类名搜索、查看包、介绍、构造器和方法。方法命名能暗示其功能,注意参数和返回值。`String`创建有两种方式:双引号创建(常量池,共享)和构造器`new`(每次新建对象)。此外,列举了`String`的常用方法,如`length()`、`charAt()`、`equals()`、`substring()`等。
15 0
|
1月前
|
Java
【Java】如果一个集合中类型是String如何使用拉姆达表达式 进行Bigdecimal类型计算?
【Java】如果一个集合中类型是String如何使用拉姆达表达式 进行Bigdecimal类型计算?
25 0
|
1月前
|
Java
Java String split()方法详细教程
Java String split()方法详细教程
23 0
|
1月前
|
安全 Java
Java StringBuffer 和 StringBuilder 类
Java StringBuffer 和 StringBuilder 类
16 0
|
1天前
|
存储 缓存 Java
|
2天前
|
存储 编解码 算法
Java 的 String StringBuilder StringBuffer(上)
Java 的 String StringBuilder StringBuffer
24 0
|
1月前
|
Java 索引
【Java】String类常用方法总结
【Java】String类常用方法总结
20 0
|
1月前
|
SQL Java
使用java中的String类操作复杂的字符串
使用java中的String类操作复杂的字符串
9 0
|
1月前
|
存储 Java 索引
Java String类
Java String类
12 0
|
1月前
|
安全 Java 索引
Java StringBuffer 类
Java StringBuffer 类
7 0