`
zhsx2221
  • 浏览: 37385 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

JVM对String对象的处理

阅读更多
java 代码
  1.   
  2. /**  
  3.  *   
  4.  * java JVM 对String对象的处理  
  5.  * test String. intern() 方法   
  6.  *   
  7.  */  
  8. public class TestStringIntern {   
  9.   
  10.   
  11.     /**  
  12.      * @param args  
  13.      */  
  14.     public static void main(String[] args) {   
  15.            
  16.         /*  
  17.          * 1.字面上的 "Hi" 字符串(字符串字面值string literal)将被自动 intern到虚拟机的字符串池中.  
  18.          * 2.不是字面上的String对象 通过调用 intern()方法,被intern到虚拟机的字符串池中.  
  19.          *   
  20.          * 虚拟机的字符串池由类 String 私有地维护。  
  21.          */  
  22.         String a = "Hi";   
  23.         String b = "Hi";   
  24.         String c = new String("Hi");   
  25.   
  26.         System.out.println(a == b);// true   
  27.         System.out.println(a == c);// false   
  28.            
  29.         c = c.intern();   
  30.         System.out.println(a == c);// true   
  31.            
  32.         /*  
  33.          * 创建了两个对象   
  34.          * 1.字面"AA" ——被intern到虚拟机字符串池中了。  
  35.          * 2.new 操作符创建的 对象  
  36.          *   
  37.          * 由AA != AA.intern()可以看出:String AA = new String("AA");创建了两个对象。  
  38.          */  
  39.         String AA = new String("AA");   
  40.         System.out.println(AA == "AA");// false   
  41.         System.out.println(AA == AA.intern());// false   
  42.         System.out.println("AA" == AA.intern());// true   
  43.            
  44.            
  45.         /*  
  46.          * JAVA 规范中的例子:  
  47.          */  
  48.         String hello = "Hello", lo = "lo";   
  49.         System.out.print((hello == "Hello") + " ");      //true   
  50.         System.out.print((hello == ("Hel"+"lo")) + " "); //true   
  51.         System.out.print((hello == ("Hel"+lo)) + " ");   //false--lo为非literal的   
  52.         System.out.println(hello == ("Hel"+lo).intern());//true   
  53.   
  54.     }   
  55.   
  56. }   
  57.   
分享到:
评论
14 楼 hbcui1984 2007-04-16  
skydream 写道
对这个的解释,我有不同的想法
"Hel"+"lo"这个表达式,编译器是有优化的,因为string是不可变的,因此"Hel","lo"都是常量,"Hel"+"lo"是常量表达式。java语言规范中常量表达式是在编译时进行求值的。因此对于"Hel"+"lo"被编译器直接处理为"Hello"。

总之,先是编译器先进行特殊处理,然后才是jvm处理,依据楼主说明的处理方式,最后得出一个true.

正解
13 楼 shaobin0604 2007-04-15  
skydream 写道
对这个的解释,我有不同的想法
#         System.out.print((hello == "Hello") + " ");      //true  
#         System.out.print((hello == ("Hel"+"lo")) + " "); //true 
hello == ("Hel"+"lo")的结果为true,没有疑义。但是为什么是true,我认为这里应该增加一个注释,才能准确的说明问题,即"Hel"+"lo"这个表达式,编译器是有优化的,因为string是不可变的,因此"Hel","lo"都是常量,"Hel"+"lo"是常量表达式。java语言规范中常量表达式是在编译时进行求值的。因此对于"Hel"+"lo"被编译器直接处理为"Hello"。

总之,先是编译器先进行特殊处理,然后才是jvm处理,依据楼主说明的处理方式,最后得出一个true.
楼上的正解
12 楼 skydream 2007-01-18  
对这个的解释,我有不同的想法
#         System.out.print((hello == "Hello") + " ");      //true  
#         System.out.print((hello == ("Hel"+"lo")) + " "); //true 
hello == ("Hel"+"lo")的结果为true,没有疑义。但是为什么是true,我认为这里应该增加一个注释,才能准确的说明问题,即"Hel"+"lo"这个表达式,编译器是有优化的,因为string是不可变的,因此"Hel","lo"都是常量,"Hel"+"lo"是常量表达式。java语言规范中常量表达式是在编译时进行求值的。因此对于"Hel"+"lo"被编译器直接处理为"Hello"。

总之,先是编译器先进行特殊处理,然后才是jvm处理,依据楼主说明的处理方式,最后得出一个true.
11 楼 zhsx2221 2006-12-29  
aone 写道
System.out.print((hello == ("Hel"+lo)) + " ");   //false--lo为非literal的
不是太明白这一句,能不能解释一下?


首先解释一下 “那张表”不是HASHTABLE,而是JVM内部维护的常量池;
对于 System.out.print((hello == ("Hel"+lo)) + " ");   这一句 要和
     System.out.print((hello == ("Hel"+"lo")) + " "); 来对照解释 :

在以上两句中("Hel"+lo) 创建了新的对象 而 ("Hel"+"lo")没有创建新的对象,而是从JVM字符串常量池中获取之前已经拘留了的"Hello"对象。

actually ,a string-valued constant expression,formed from two string literals。
同样它首先会从JVM字符串常量池中查找相应的 String对象(如“Hello”),如果找到则直接获取引用,如果不存在那么它也会被JVM拘留/intern 。

对于("Hel"+lo)创建了新的对像是因为:lo 是一个变量/引用,它并不是字面上的字符串(string literal),同理这个 表达式 也并不是一个string-valued constant expression,所以他不会被拘留,而是创建 一个新的对象。

---------
越说越绕,希望能理解......


10 楼 wjtang 2006-12-22  
get
9 楼 aone 2006-12-22  
System.out.print((hello == ("Hel"+lo)) + " ");   //false--lo为非literal的
不是太明白这一句,能不能解释一下?
8 楼 dwangel 2006-12-22  
galaxystar 写道
ahuaxuan 写道
事实上,每一个jvm必须维护一张内部表,它会列出所有在程序运行的过程中被拘留的字符串对象引用。基本上,如果一个字符串在jvm的这张内部表上出现,就是说它是被jvm所拘留的。维护这个列表的关键是任何特定的字符序列在这个列表上只出现一次。比如说上面的"hello",被拘留之后,虚拟机会使用指向以前拘留的字符串对象的引用,也就是说引用a和引用b是一个引用,这也解释了为什么a==b是为true的


a == b,记得是用hashCode比较的!你说的那张内部表,是指HashTable吗?

是比较a跟b这个变量里的内存地址不是hashCode。
7 楼 liucjj 2006-12-21  
c应该是引用的内存地址,那么a呢
6 楼 ahuaxuan 2006-12-20  
引用
galaxystar说道:

a == b,记得是用hashCode比较的!你说的那张内部表,是指HashTable吗?



我想这个内部表是jvm内部的表示,而jvm是c写的,是不是HashTable我就不知道了,c我不熟,hashcode 本身也是一个jni调用,如果对jvm不是非常熟的话,很难知道jvm是怎么做的
5 楼 galaxystar 2006-12-20  
ahuaxuan 写道
事实上,每一个jvm必须维护一张内部表,它会列出所有在程序运行的过程中被拘留的字符串对象引用。基本上,如果一个字符串在jvm的这张内部表上出现,就是说它是被jvm所拘留的。维护这个列表的关键是任何特定的字符序列在这个列表上只出现一次。比如说上面的"hello",被拘留之后,虚拟机会使用指向以前拘留的字符串对象的引用,也就是说引用a和引用b是一个引用,这也解释了为什么a==b是为true的


a == b,记得是用hashCode比较的!你说的那张内部表,是指HashTable吗?
4 楼 zhsx2221 2006-12-20  
用例子说话,再加上相关的注释 会比白话描述更清晰 hehe
3 楼 gao277 2006-12-15  
以前觉得自己挺明白这个东西的,看了文章以后更加明白了。
2 楼 ahuaxuan 2006-12-15  
事实上,每一个jvm必须维护一张内部表,它会列出所有在程序运行的过程中被拘留的字符串对象引用。基本上,如果一个字符串在jvm的这张内部表上出现,就是说它是被jvm所拘留的。维护这个列表的关键是任何特定的字符序列在这个列表上只出现一次。比如说上面的"hello",被拘留之后,虚拟机会使用指向以前拘留的字符串对象的引用,也就是说引用a和引用b是一个引用,这也解释了为什么a==b是为true的
1 楼 icefire 2006-12-08  
String a = "Hi";   
String b = "Hi";

系统对所有的字符串常量和常值的String表达式都自动进行扣留。
因为String类型对象被称作是永久的.

相关推荐

    String创建对象

    // 转换为String对象 ``` 4. **intern() 方法**: `String`类的`intern()`方法用于获取常量池中的字符串引用。如果常量池中已存在该字符串,则返回其引用;否则,将字符串添加到常量池并返回引用。例如: ```...

    String及StringBuffer和StringBuilder的区别

    这意味着,如果经常改变字符串内容,最好不要使用 String 类型,因为每次生成对象都会对系统性能产生影响,特别是在内存中无引用对象多了以后,JVM 的 GC 就会开始工作,那速度一定会相当慢的。 StringBuffer ...

    Java JVM Instruction Set

    这有助于更深入地理解JVM如何处理不同类型的对象引用和方法分派。 综上所述,Java虚拟机指令集是Java程序执行的关键组成部分。通过深入研究JVM的工作原理和指令集,可以帮助我们更好地理解和优化Java程序的性能。

    JVM 深入学习教程深入分析JVM教程!jvm 内存原型,优化等等

    2. 常见泄漏原因:静态集合类引用对象,单例模式未正确处理,长生命周期对象持有短生命周期对象等。 六、类加载机制 1. 双亲委派模型:类加载请求先向上委托给父加载器,只有当父加载器无法加载时,才由当前加载器...

    对String的深入理解

    当使用`+`进行字符串连接时,JVM会自动处理字符串转换和连接过程。对于`str = 1+2+str+3+4;`这样的表达式,每次`+`运算都会生成一个新的`String`对象。在这个例子中,会产生5个对象:原始的"jf",数字1转换的字符串...

    String、StringBuilder和StringBuffer的区别

    这意味着每次对String对象进行修改(如拼接操作),实际上都会创建一个新的String对象,这在频繁修改字符串内容时可能会导致大量的内存开销,特别是在多线程环境中,这种性能问题更加显著。例如,以下代码中,每进行...

    jvm8虚拟机规范

    JVM8引入了G1垃圾收集器、String去重复、堆内存并行压缩等优化措施,提升了内存管理和性能。同时,还引入了 invokedynamic 指令,支持动态语言的高效运行。 6. 类加载机制: 类加载分为加载、验证、准备、解析和...

    JVM性能调优-JVM内存整理及GC回收.pdf_java_jvm_

    7. **字符串池优化**:使用String对象池,减少对象创建,提升性能。 总结来说,JVM性能调优是多方面的工作,包括理解内存结构、选择合适的垃圾收集器、调整内存参数以及优化代码实现。通过细致入微的调优,可以显著...

    StringTable.pdf

    这意味着每次对字符串的修改(如拼接、替换等)都会生成新的String对象。 4. **内存优化**:JDK9之后,String的内部表示由`final char[] value`改为`final byte[]`,减少了内存消耗,特别是对于拉丁字符,只需一个...

    JVM性能调优-JVM内存整理及GC回收.pdf

    这就解释了为何在方法内部对对象状态的改变会影响原始对象的状态。 例如,考虑以下代码片段: ```java public class Main { public static void main(String[] args) { int x = 10; Integer y = new Integer(20...

    java虚拟机JVM详解ppt

    通过对Java虚拟机JVM的基本概念、架构以及内存管理机制的深入解析,我们可以更好地理解Java程序的执行过程。掌握JVM的工作原理有助于开发者编写更高效、更稳定的Java应用程序,并能有效地解决常见的内存溢出问题。

    怎样用Jvm处理Java数组.doc

    这种方式避免了对标准库的污染,并允许JVM根据实际需求灵活处理各种类型的数组。 4. **数组长度字段的特殊处理** 尽管反射技术显示数组类并没有直接声明`length`成员变量,但这并不意味着数组对象中没有存储长度...

    Java 中的 String对象为什么是不可变的

    在Java编程语言中,String对象被认为是不可变的,这意味着一旦创建了一个String实例,就不能更改它的内容。这种特性是由String类的设计决定的,它对程序员在处理字符串时提供了许多优势和安全保证。 首先,理解什么...

    Java进阶教程解密JVM视频教程

    * 在垃圾回收章节,不仅会介绍垃圾回收算法、分代垃圾回收机制,还会重点介绍 G1 垃圾回收器,辨析 Full GC 发生条件,jdk8以来对垃圾回收的优化,以及垃圾回收的调优法则。 * 在字节码与类加载技术章节,会从一个 ...

    JVM的垃圾回收机制详解和调优

    8. **代码优化**:减少不必要的对象创建,如避免使用大量的临时对象,合理使用集合类的容量,以及使用String的concat()方法代替+操作符构建字符串。 9. **使用最新JVM版本**:新版本的JVM通常包含性能优化和新的GC...

    Java 6 JVM参数选项大全

    新生代收集担保是确保在Minor GC期间,如果新生代的对象无法全部转移到Survivor区,JVM有足够的空间在老年代进行分配。关闭此功能可能导致在某些情况下更频繁的Full GC,但可以减少内存浪费和潜在的性能提升。 在...

    深入理解JVM & G1 GC

    3. 年老代优先:G1会优先处理年老代的垃圾,因为这些对象通常更大,回收效果更明显。 4. 空间整理:G1会尝试避免碎片,通过整体移动存活对象来整理空间。 G1 GC的调优主要包括以下几个方面: 1. 设置合适的堆大小...

    java String类的实现

    Java虚拟机(JVM)维护了一个字符串常量池,用于存储所有的字面量字符串和通过`new String()`创建的字符串。如果字符串已经存在于池中,`String`构造函数将不会创建新对象,而是返回池中已有的引用。 ### 7. `...

Global site tag (gtag.js) - Google Analytics