1.String类概念

(1)String是final的,不可被继承。public final class String。String是的本质是字符数组char[], 并且其值不可改变。private final char value[];

(2)Java运行时会维护一个String Pool(String池)。String池用来存放运行时中产生的各种字符串,并且池中的字符串的内容不重复。而一般对象不存在这个缓冲池,仅仅存在于方法的堆栈区。

(3)创建字符串的方式很多,归纳起来有三类:(1)使用new关键字创建字符串-->String s1 = new String("abc");(2)直接指定-->String s2 = "abc";(3)用串联生成新的字符串-->String s3 = "ab" + "c";

2.String对象创建的机制

原理1:当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个s在String池中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。例如:String str="abc";这行代码被执行的时候,JAVA虚拟机首先在字符串池中查找是否已经存在了值为"abc"的这么一个对象,它的判断依据是String 类equals(Object obj)方法的返回值。如果有,则不再创建新的对象,直接返回已存在对象的引用;如果没有,则先创建这个对象,然后把它加入到字符串池中,再将它的引用返回。

原理2:Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。而不在常量池中创建String对象,只有使用了str.intern(),才会在常量池中创建一个String对象。

原理3:使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。

原理4:使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。

最后,有几点问题请大家注意:String a; 与String a=null在作为类变量时候是等价的,在局部变量则不同。null表示一个空引用,String a=null意思是在栈中声明了a,但是这个a没有指向任何地址。此时我们注意到String a在栈中声明了a,但是也没有指向任何地址,但是java的语法检查如果在局部变量中,String a是不能直接使用的,String a=null中的这个a可以直接使用。

3.经典面试题

(1)String s = new String("abc");创建了几个String Object? 答:[两个,pool中1个,heap中1个]

(2)String s0 = new String("abc");String s1 = new String("abc");创建了几个String Object?答:[三个,pool中1个,heap中2个]

(3)解释

涉及概念:字符串池[pool of literal strings]的字符串对象和堆[heap]中的字符串对象。

字符串对象的创建:由于字符串对象的大量使用[它是一个对象,一般而言普通对象总是在heap分配内存],Java中为了节省内存空间和运行时间,在编译阶段就把所有的字符串文字放到一个字符串池[pool of literal strings]中,而运行时字符串池成为常量池的一部分。字符串池的好处,就是该池中所有相同的字符串常量被合并,只占用一个空间。另一种解释:在JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象,并且可以被共享使用,因此它提高了效率。由于String类是final的,它的值一经创建就不可改变,因此我们不用担心String对象共享而带来程序的混乱。字符串池由String类维护,我们可以调用intern()方法来访问字符串池。

现在看String s = new String("abc")语句,在执行new String()时,先检查pool中有没有"abc"对象,如果没有,则先在pool中创建一个"abc"对象,再将其复制一份放到heap中,并且把heap中的这个对象的引用交给s持有,这条语句就创建了2个String对象;如果有则将pool中的对象复制一份放到heap中,并且把heap中的这个对象的引用交给s持有,这条语句就创建了1个String对象(也可以理解为:如果有则新创建的"abc"对象将原有pool中的"abc"对象覆盖,这样可以勉强说创建了2个String对象)。

补充:常量池(constant pool)指的是在编译期被确定,并被保存在已编译的.class文件中的一些数据。它包括了关于类、方法、接口等中的常量,也包括字符串常量。jdk编译时会将字符串池并入常量池。

4.举例

  1. public static void main(String[] args) {
  2. String str_0 = "forrest";
  3. String str_1 = "forrest";
  4. if (str_0 == str_1) System.out.println("pool中值创建了一个String对象:/"forrest/", str_0和str_1分别对其进行了引用");
  5. else System.out.println("Trouble");
  6. String str_2 = "vivian";
  7. String str_3 = new String("vivian");
  8. String str_4 = new String("vivian");
  9. if (str_2 != str_3) System.out.println("str_2是对pool中String对象引用, str_3是对heap中String对象引用, 所以二者不是引用的同一个对象");
  10. else System.out.println("Trouble");
  11. if (str_2 != str_4) System.out.println("str_2是对pool中String对象引用, str_4是对heap中String对象引用, 所以二者不是引用的同一个对象");
  12. else System.out.println("Trouble");
  13. if (str_3 != str_4) System.out.println("str_3和str_4都是对heap中String对象引用, 但二者引用对象的内存地址不同, 所以二者不是引用的同一个对象");
  14. else System.out.println("Trouble");
  15. String str_5 = str_3.intern(); //把str_3在heap中的String对象复制到loop中(虽然在String str_3 = new String("vivian")时已经在loop中创建了一份, 但intern()方法会要求重新复制),并将loop中这个对象引用到str_5
  16. if (str_5 != str_3) System.out.println("str_5是对pool中String对象引用, str_3是对heap中String对象引用, 所以二者不是引用的同一个对象");
  17. else System.out.println("Trouble");
  18. if (str_5 == str_2) System.out.println("str_5和str_2都是对loop中某一String对象引用, 所以二者引用的是同一个对象");
  19. else System.out.println("Trouble");
  20. str_4.intern();
  21. if (str_4 != str_2) System.out.println("str_4虽然执行了str_4.intern(), 但它的返回值没有赋给str_4, 所以str_4和str_2不是引用的同一个对象");
  22. else System.out.println("Trouble");
  23. if (str_4.intern() == str_2) System.out.println("str_4.intern()和str_2都是对loop中某一String对象引用, 所以二者引用的是同一个对象");
  24. else System.out.println("Trouble");
  25. if (str_4.intern() != str_4) System.out.println("str_4.intern()是对pool中String对象引用, str_4是对heap中String对象引用, 所以二者不是引用的同一个对象");
  26. else System.out.println("Trouble");
  27. }
  28. /*
  29. * 输出为:
  30. * pool中值创建了一个String对象:"forrest", str_0和str_1分别对其进行了引用
  31. * str_2是对pool中String对象引用, str_3是对heap中String对象引用, 所以二者不是引用的同一个对象
  32. * str_2是对pool中String对象引用, str_4是对heap中String对象引用, 所以二者不是引用的同一个对象
  33. * str_3和str_4都是对heap中String对象引用, 但二者引用对象的内存地址不同, 所以二者不是引用的同一个对象
  34. * str_5是对pool中String对象引用, str_3是对heap中String对象引用, 所以二者不是引用的同一个对象
  35. * str_5和str_2都是对loop中某一String对象引用, 所以二者引用的是同一个对象
  36. * str_4虽然执行了str_4.intern(), 但它的返回值没有赋给str_4, 所以str_4和str_2不是引用的同一个对象
  37. * str_4.intern()和str_2都是对loop中某一String对象引用, 所以二者引用的是同一个对象
  38. * str_4.intern()是对pool中String对象引用, str_4是对heap中String对象引用, 所以二者不是引用的同一个对象
  39. *
  40. * */
  1. public static void main(String[] args) {
  2. String a = "ab";
  3. String b = "cd";
  4. String c = "abcd";
  5. String d = "ab" + "cd";
  6. // 如果d和c指向了同一个对象,则说明d被加入字符串池
  7. if (c == d) System.out.println("/"ab/"+/"cd/" 创建的对象 /"加入了/" 字符串池中");
  8. else System.out.println("/"ab/"+/"cd/" 创建的对象 /"没加入/" 字符串池中");
  9. String e = a + "cd";
  10. // 如果e和c指向了同一个对象,则说明e也被加入了字符串池
  11. if (e == c) System.out.println("a +/"cd/" 创建的对象 /"加入了/" 字符串池中");
  12. else System.out.println("a +/"cd/" 创建的对象 /"没加入/" 字符串池中");
  13. String f = "ab" + b;
  14. // 如果f和c指向了同一个对象,则说明f也被加入了字符串池
  15. if (f == c) System.out.println("/"ab/"+ b 创建的对象 /"加入了/" 字符串池中");
  16. else System.out.println("/"ab/"+ b 创建的对象 /"没加入/" 字符串池中");
  17. String g = a + b;
  18. // 如果g和c指向了同一个对象,则说明g也被加入了字符串池
  19. if (g == c) System.out.println("a + b 创建的对象 /"加入了/" 字符串池中");
  20. else System.out.println("a + b 创建的对象 /"没加入/" 字符串池中");
  21. }
  22. /*
  23. * 输出为:
  24. * "ab"+"cd" 创建的对象 "加入了" 字符串池中
  25. * a +"cd" 创建的对象 "没加入" 字符串池中
  26. * "ab"+ b 创建的对象 "没加入" 字符串池中
  27. * a + b 创建的对象 "没加入" 字符串池中
  28. *
  29. * */

从第二段代码结果中我们不难看出,只有使用引号包含文本的方式创建的String对象之间使用“+”连接产生的新对象才会被加入字符串池中。对于所有包含new方式新建对象(包括null)的“+”连接表达式,它所产生的新对象都不会被加入字符串池中;对于类似str=1+"AAA"+2+3这种强制类型转化形成的String对象也不会放到pool中的。对此我们不再赘述。因此我们提倡大家用引号包含文本的方式来创建String对象以提高效率,实际上这也是我们在编程中常采用的。

最新文章

  1. 【Bootstrap Demo】入门例子创建
  2. 正在编译转换: 未能找到元数据文件 EntityFramework.dll
  3. 工作的思考十五:升职前需要做的准备(TeamLeader)
  4. ios UICollectionView滑动时操作
  5. Cocos2d-x 3.x 资料整理
  6. Java write And read Demo
  7. Gmail 一些需注意的事
  8. 教你使用vim表白
  9. Reverse Words in a String III
  10. 笔记:Hibernate DML
  11. Vcenter 账户密码过期设置修改
  12. Kaggle-房价预测
  13. Leetcode 349. 两个数组的交集 By Python
  14. NEST - 编写查询
  15. [pytorch修改]npyio.py 实现在标签中使用两种delimiter分割文件的行
  16. LinkedList 的源码分析
  17. 浅析Java 数组-基础详解
  18. Oracle 截取字符串(截取固定分隔符中间的字符
  19. maven 之dependencyManagement 和 pluginManagement
  20. 详解Base64编码和解码

热门文章

  1. UI4_UIToolBar
  2. ES6新增Promise
  3. 济南学习 Day2 T1 am
  4. unity 3消 游戏
  5. Windows phone 中一些实用的控件
  6. 《shell下sort排序命令的使用》
  7. silverlight 用户浏览器未安装SL插件问题
  8. js中substring和substr的用法比较
  9. jQuery对Select操作大集合
  10. Linux使用标准IO的调用函数,分3种形式实现