java中字符串的初始化过程

java中字符串的初始化过程以及String Stringbuilder StringBuffer 的区别:

在Java的JVM中,有一个字符串常量池的概念,在jdk 1.7之后,字符串常量池被划分到java的Heap中,Java的八种基本数据类型中,除了float 和 Double ,其他都实现了常量池技术。

字符串常量池底层是用HashTable实现的,以  key---value  的形式存储数据

1.  String  s1 = "hello world";    String  s2 = new String("hello world"); 有什么区别呢?

      System.out.println(s1 == s2); 打印的结果是True 还是  False ?

结论:使用双引号的形式直接声明字符串,在JVM 中的操作过程是:(1)JVM首先会到字符串常量池中查找有没有“hello   world”字   符串,如果有,那么直接引用常量池中的。(2)如果没有,那么JVM首先会在堆内存(Heap)中new一个字符串对象,然后在常量池中注册一个该字符串。

       使用new关键字创建一个字符串,在JVM 中的操作过程是:JVM不会查询字符串常量池,会直接在Heap中创建一个新的对        象。

所以上面打印的结果为False

 

2.  String  s1 = new String("hello") + new String("world");

     s1.inter();

     String  s2 = "hello world";

     System.out.println(s1 == s2);

    该结果输出为True,当s1调用inter();时,JVM首先会去常量池中查找是否有该字符串的引用,如果有,直接返回该引用,

如果没有,JVM会将s1字符串的引用注册到常量池中,当s2创建时,会从常量池中找得到,所以s2与s1 相等。

 

3.   String  s1 = "hello";  String  s2 = "world";

      String   s3 = s1 + s2;  String  s4 = "hello world";

      System.out.println(s3 == s4);

     解决这个问题的关键是要判断s3中的s1+s2是以双引号的形式创建,还是以new的形式创建。使用  javap  -c  反编译代码,可以看出,两个字符串相加,底层使用的是StringBuilder.append(); ,然后会以StringBuilder.toString()的形式输出,在toString();中,使用了 new String对象的形式返回一个字符串,所以s3是通过new 的形式来创建的,JVM不会将 s3  写入常量池中,所以 s4 也就不会从常量池中拿到 "hello world"字符串,所以输出的结果为false。

 

4.String , StringBuffer , StringBuilder 的底层实现:

(1) String ,StringBuffer , StringBuilder 底层都是采用 char 类型的数组实现。

(2) String 为不可变类型的字符串,在String 类中,所有返回值类型为String类型的方法中,都会 new 一个新的字符串。

     StringBuffer 和 StringBuilder 会使用append();在现有的字符串基础上拼接得到。但是StringBuffer类中的方法,添加了

     Synchronized关键字,它是线程安全的,所以效率会比StringBuilder低。

(3)  使用String 类型进行字符串的拼接时,每一次拼接都会生成一个StringBuilder对象,调用append(); 然后在调用 toString(),

      通过new 一个新的字符串的形式来返回。在进行多次拼接时,效率会很低。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值