String分析

  • 栈(stack):主要保存基本类型(或者叫内置类型)(char、byte、short、int、long、float、double、boolean)和对象的引用,数据可以共享,速度仅次于寄存器(register),快于堆。
  • 堆(heap):用于存储对象。 
  • String池(pool) :JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象; 并且可以被共享使用,因此它提高了效率。

    代码解析:

    [java]  view plain copy
    1. public static void main(String[] args) {  
    2.         /** 
    3.          * 情景一:字符串池 
    4.          * JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多String对象; 
    5.          * 并且可以被共享使用,因此它提高了效率。 
    6.          * 由于String类是final的,它的值一经创建就不可改变。 
    7.          * 字符串池由String类维护,我们可以调用intern()方法来访问字符串池。  
    8.          */  
    9.         String s1 = "abc";     
    10.         //↑ 在字符串池创建了一个对象  
    11.         String s2 = "abc";     
    12.         //↑ 字符串pool已经存在对象“abc”(共享),所以创建0个对象,累计创建一个对象  
    13.         System.out.println("s1 == s2 : "+(s1==s2));    
    14.         //↑ true 指向同一个对象,  
    15.         System.out.println("s1.equals(s2) : " + (s1.equals(s2)));    
    16.         //↑ true  值相等  
    17.         //↑------------------------------------------------------over  
    18.         /** 
    19.          * 情景二:关于new String("") 
    20.          *  
    21.          */  
    22.         String s3 = new String("abc");  
    23.         //↑ 创建了两个对象,一个存放在字符串池中,一个存在与堆区中;  
    24.         //↑ 还有一个对象引用s3存放在栈中  
    25.         String s4 = new String("abc");  
    26.         //↑ 字符串池中已经存在“abc”对象,所以只在堆中创建了一个对象  
    27.         System.out.println("s3 == s4 : "+(s3==s4));  
    28.         //↑false   s3和s4栈区的地址不同,指向堆区的不同地址;  
    29.         System.out.println("s3.equals(s4) : "+(s3.equals(s4)));  
    30.         //↑true  s3和s4的值相同  
    31.         System.out.println("s1 == s3 : "+(s1==s3));  
    32.         //↑false 存放的地区多不同,一个栈区,一个堆区  
    33.         System.out.println("s1.equals(s3) : "+(s1.equals(s3)));  
    34.         //↑true  值相同  
    35.         //↑------------------------------------------------------over  
    36.         /** 
    37.          * 情景三:  
    38.          * 由于常量的值在编译的时候就被确定(优化)了。 
    39.          * 在这里,"ab"和"cd"都是常量,因此变量str3的值在编译时就可以确定。 
    40.          * 这行代码编译后的效果等同于: String str3 = "abcd"; 
    41.          */  
    42.         String str1 = "ab" + "cd";  //1个对象  
    43.         String str11 = "abcd";   
    44.         System.out.println("str1 = str11 : "+ (str1 == str11));  
    45.         //↑------------------------------------------------------over  
    46.         /** 
    47.          * 情景四:  
    48.          * 局部变量str2,str3存储的是存储两个拘留字符串对象(intern字符串对象)的地址。 
    49.          *  
    50.          * 第三行代码原理(str2+str3): 
    51.          * 运行期JVM首先会在堆中创建一个StringBuilder类, 
    52.          * 同时用str2指向的拘留字符串对象完成初始化, 
    53.          * 然后调用append方法完成对str3所指向的拘留字符串的合并, 
    54.          * 接着调用StringBuilder的toString()方法在堆中创建一个String对象, 
    55.          * 最后将刚生成的String对象的堆地址存放在局部变量str3中。 
    56.          *  
    57.          * 而str5存储的是字符串池中"abcd"所对应的拘留字符串对象的地址。 
    58.          * str4与str5地址当然不一样了。 
    59.          *  
    60.          * 内存中实际上有五个字符串对象: 
    61.          *       三个拘留字符串对象、一个String对象和一个StringBuilder对象。 
    62.          */  
    63.         String str2 = "ab";  //1个对象  
    64.         String str3 = "cd";  //1个对象                                         
    65.         String str4 = str2+str3;                                        
    66.         String str5 = "abcd";    
    67.         System.out.println("str4 = str5 : " + (str4==str5)); // false  
    68.         //↑------------------------------------------------------over  
    69.         /** 
    70.          * 情景五: 
    71.          *  JAVA编译器对string + 基本类型/常量 是当成常量表达式直接求值来优化的。 
    72.          *  运行期的两个string相加,会产生新的对象的,存储在堆(heap)中 
    73.          */  
    74.         String str6 = "b";  
    75.         String str7 = "a" + str6;  
    76.         String str67 = "ab";  
    77.         System.out.println("str7 = str67 : "+ (str7 == str67));  
    78.         //↑str6为变量,在运行期才会被解析。  
    79.         final String str8 = "b";  
    80.         String str9 = "a" + str8;  
    81.         String str89 = "ab";  
    82.         System.out.println("str9 = str89 : "+ (str9 == str89));  
    83.         //↑str8为常量变量,编译期会被优化  
    84.         //↑------------------------------------------------------over  
    85.     }  

    总结:
    1、当使用任何方式来创建一个字符串对象s时,Java运行时(运行中JVM)会拿着这个X在String池中找是否存在内容相同的字符串对象,如果不存在,则在池中创建一个字符串s,否则,不在池中添加。
    2、Java中,只要使用new关键字来创建对象,则一定会(在堆区或栈区)创建一个新的对象。 
    3、使用直接指定或者使用纯字符串串联来创建String对象,则仅仅会检查维护String池中的字符串,池中没有就在池中创建一个,有则罢了!但绝不会在堆栈区再去创建该String对象。 
    4、使用包含变量的表达式来创建String对象,则不仅会检查维护String池,而且还会在堆栈区创建一个String对象。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值