Java中String类因为其特殊性(不变类),几乎是笔试面试中的必考题,固然有些题目其实没啥意思,不过关键是要经过题目掌握原理性的东西。下面六道题目,若是您所有作对了,且明白其因此然,那么Java中的关于String的笔试面试题应该难不到你了。也许您以为polaris说的有点过了,然而完全明白这些题目,对理解String类仍是颇有好处的。java
写出下面各题的打印输出的结果:面试
1 设计模式
2数组
3函数
4源码分析
5设计
6对象
(1)经过java源码分析String继承
咱们都知道String是不可变的(immutable),不变性的体现是:String类内部经过char数组来保存字符串,而这个char数组被声明为:final。那么为何要将String类声明为不可变呢?了解设计模式的你应该知道有一种模式叫作“不变模式”(immutable pattern),String类的设计就是使用了不变模式,有兴趣的朋友能够看看“不变模式”讲的具体是啥东东。索引
说完String的不可变性,须要说说String的“final性”(其实也仍是不可变性决定的)。这也是有些面试官会问到的问题:我能不能写一个类继承自String?为何?咱们来看看String类的声明:
public final class String implements java.io.Serializable, Comparable<String>, CharSequence
对于final关键字的做用不用多解释了。其实这也是“强不变模式”的一种要求(类自己声明为final或全部方法声明为final)。
(2)理解String对象的存储机制
要深刻理解String必须先了解Java内存机制——运行时数据区(Runtime Data Area)。《The JavaTM Virtual Machine Specification》中将运行时数据区分为六部分(参看第三章): 1)The pc Register;2)Java Virtual Machine Stacks;3)Heap;4)Method Area;5)Runtime Constant Pool;6)Native Method Stacks; 以上数据区的具体描述可参考规范。须要注意的是,以上只是一个规范说明,并无规定虚拟机如何实现这些数据区。
在说明String对象存储机制以前,咱们须要先了解数据区的三个部分:Java 虚拟机栈(能够简称为Java栈)、堆和运行时常量池(简称常量池)。对于Java栈和堆你们应该比较熟悉,这里有一个关键点是常量池,下面就重点介绍一下与String相关的常量池。
首先大概描述一下什么是常量池:
虚拟机必须为每一个被装载的类型维护一个常量池。常量池就是该类型所用常量的一个有序集合,包括直接常量(string,integer和floating point常量)和对其余类型、字段和方法的符号引用。池中的数据项就像数组同样是经过索引访问的。由于常量池存储了相应类型所用到的全部类型、字段和方法的符号引用,因此它在Java程序的动态连接中起着核心的做用。
<1> String相关常量池
在《The JavaTM Virtual Machine Specification》第四章有一节是专门讲解各类常量池的,其中有两个常量池是关于String的。
1)The CONSTANT_String_info Structure
对于常量池的细节此文不作过多介绍,polaris之后可能会写一序列关于Java虚拟机的文章。如今您能够查阅规范或在网上收集相关资料阅读。规范上对该常量池结构的介绍是: The CONSTANT_String_info structure is used to represent constant objects of the type String. 在该常量池结构中引用了另外一个常量池结构,如2)
2)The CONSTANT_Utf8_info Structure
规范上的描述是:The CONSTANT_Utf8_info structure is used to represent constant string values.
根据上面的介绍能够看出,字符串字面值会存储在常量池中。下面来分析String对象的存储机制。
<2> String对象的存储
请看这样两个语句:
String x = "abc"; String y = new String("abcd");
如今来分析一下内存的分配状况。如图:
能够看出,x与y存在栈中,它们保存了相应对象的引用。第一条语句没有在堆中分配内存,而是将“abc”保存在常量池中。对于第二条语句,一样会在常量池中有一个“abcd”的字符串,当new时,会拷贝一份该字符串存放到堆中,因而y指向了堆中的那个“abcd”字符串。不知道polaris有没有讲明白。若是您明白了,那么作前面那六道题就没什么问题了。
三、六道题答案详解
1)true
要说明一点:当两个字符串字面值链接时(相加),获得的新字符串依然是字符串字面值,保存在常量池中。
2)false
当字符串字面值与String类型变量链接时,获得的新字符串再也不保存在常量池中,而是在堆中新建一个String对象来存放。很明显常量池中要求的存放的是常量,有String类型变量固然不能存在常量池中了。
3)true
注意此题与上一题的区别,此处是字符串字面值与String类型常量链接,获得的新字符串依然保存在常量池中。
4)false
此题中第条语句:final String bb = getBB();其实与final String bb = new String(“b”);是同样的。也就是说return “b”会在堆中建立一个String对象保存”b”,虽然bb被定义成了final。可见并不是定义为final的就保存在常量池中,很明显此处bb常量引用的String对象保存在堆中,由于getBB()获得的String已经保存在堆中了,final的String引用并不会改变String已经保存在堆中这个事实。
5)false,true
可能不少人对intern()这个函数不了解。JDK API文档中对intern()方法的描述是:
返回字符串对象的规范化表示形式。
一个初始为空的字符串池,它由类 String 私有地维护。
当调用 intern 方法时,若是池已经包含一个等于此 String 对象的字符串(用 equals(Object) 方法肯定),则返回池中的字符串。不然,将此 String 对象添加到池中,并返回此 String 对象的引用。
它遵循如下规则:对于任意两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。
全部字面值字符串和字符串赋值常量表达式都使用 intern 方法进行操做。
上面字符串池即为字符串常量池。明白该题结果的缘由了吧。
6)false,false,true