java字符串的深入理解

  

<强> java字符串的深入理解

  

<强>一、Java内存模型,
  

  

按照官方的说法:Java虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配只,,,
  

  

JVM主要管理两种类型内存:堆和非堆,堆内存(堆内存)是在Java虚拟机启动时创建,非堆内存(非堆内存)是在JVM堆之外的内存。

  

简单来说,非堆包含方法区,JVM内部处理或优化所需的内存(如JITCompiler,即时编译器,即时编译后的代码缓存),每个类结构(如运行时常数池,字段和方法数据)以及方法和构造方法的代码只,,,,
  

  

Java的堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过新的,newarray, anewarray和multianewarray等指令建立,它们不需要程序代码来显式的释放。

  

<强>堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。

  

<强>栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性强。栈中主要存放一些基本类型的变量数据(int,短的、长的、字节、浮点数、双,布尔,字符)和对象句柄(引用)只,,,,
  

  

虚拟机必须为每个被装载的类型维护一个常量池。常量池就是该类型所用到常量的一个有序集合,包括直接常量(字符串、整数和浮点常量)和对其他类型,字段和方法的符号引用。

  

对于字符串常量,它的值是在常量池中的。而JVM中的常量池在内存当中是以表的形式存在的,对于字符串类型,有一张固定长度的CONSTANT_String_info表用来存储文字字符串值,注意:该表只存储文字字符串值,不存储符号引用。说到这里,对常量池中的字符串值的存储位置应该有一个比较明了的理解了。在程序执行的时候,常量池会储存在方法区,而不是堆中。常量池中保存着很多字符串对象;并且可以被共享使用,因此它提高了效率,
  

  

<强>二、案例解析
  

        公共静态void main (String [] args) {/* *   *情景一:字符串池   * JAVA虚拟机(JVM)中存在着一个字符串池,其中保存着很多字符串对象;   *并且可以被共享使用,因此它提高了效率。   *由于字符串类是最终的,它的值一经创建就不可改变。   *字符串池由字符串类维护,我们可以调用实习生()方法来访问字符串池。   */字符串s1=癮bc”;//谧址卮唇艘桓龆韵?   字符串s2=癮bc”;//址匾丫嬖诙韵蟆癮bc”(共享),所以创建0个对象,累计创建一个对象   system . out。println (“s1==s2:”+ (s1==s2));//嬲赶蛲桓龆韵?   System.out.println (“s1.equals (s2): " + (s1.equals (s2)));//媸抵迪嗟?/? - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -/* *   *情景二:关于新的字符串(" ")   *   */字符串s3=新的字符串(“abc”);//唇肆礁龆韵?一个存放在字符串池中,一个存在与堆区中;//褂幸桓龆韵笠胹3存放在栈中   字符串s4=新的字符串(“abc”);//址刂幸丫嬖凇癮bc”对象,所以只在堆中创建了一个对象   system . out。println (“s3==s4:”+ (s3==s4));//賡3和s4栈区的地址不同,指向堆区的不同地址;   System.out.println (“s3.equals (s4): " + (s3.equals (s4)));//嬲齭3和s4的值相同   system . out。println (“==s1:”+ (s1==s3));//俅娣诺牡厍嗖煌?一个栈区,一个堆区   System.out.println (“s1.equals (s3):”+ (s1.equals (s3)));//媸抵迪嗤?/? - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -/* *   *情景三:   *由于常量的值在编译的时候就被确定(优化)了。   *在这里,“ab”和“cd”都是常量,因此变量str3的值在编译时就可以确定。   *这行代码编译后的效果等同于:字符串str3=癮bcd”;   */字符串str1=癮b”+“cd”;//1个对象   字符串str11=癮bcd”;   system . out。println (“str1=str11:”+ (str1==str11));//? - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -/* *   *情景四:   *局部变量str2, str3存储的是存储两个拘留字符串对象(实习生字符串对象)的地址。   *   *第三行代码原理(str2 + str3):   *运行期JVM首先会在堆中创建一个StringBuilder类,   *同时用str2指向的拘留字符串对象完成初始化,   *然后调用附加方法完成对str3所指向的拘留字符串的合并,   *接着调用StringBuilder的toString()方法在堆中创建一个字符串对象,   *最后将刚生成的字符串对象的堆地址存放在局部变量str3中。   *   *而str5存储的是字符串池中“abcd”所对应的拘留字符串对象的地址。   * str4与str5地址当然不一样了。   *   *内存中实际上有五个字符串对象:   *三个拘留字符串对象,一个字符串对象和一个StringBuilder对象。   */字符串str2=癮b”;//1个对象   字符串str3=癱d”;//1个对象   字符串str4=str2 + str3;   字符串str5=癮bcd”;   system . out。println (“str4=str5:”+ (str4==str5));//错误//? - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -/* *   *情景五:   * JAVA编译器对字符串+基本类型/常量是当成常量表达式直接求值来优化的。   *运行期的两个字符串相加,会产生新的对象的,存储在堆(heap)中   */字符串str6=癰”;   字符串str7=a + str6;   字符串str67=癮b”;   system . out。println (“str7=str67:”+ (str7==str67));//黶tr6为变量,在运行期才会被解析。   最终字符串str8=癰”;   字符串str9=a + str8;   字符串str89=癮b”;   system . out。println (“str9=str89:”+ (str9==str89));//黶tr8为常量变量,编译期会被优化//? - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -   }   

java字符串的深入理解