`
longgangbai
  • 浏览: 7339560 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Java String 在JVM中的思考

阅读更多
package com;

/**
 * @author longgangbai
 * 
 */
public class StringTest {
	public static void main(String[] args) {
		String a = "ab";// 创建了一个对象,并加入字符串池中
		System.out.println("String a = \"ab\";");
		String b = "cd";// 创建了一个对象,并加入字符串池中
		System.out.println("String b = \"cd\";");
		String c = "abcd";// 创建了一个对象,并加入字符串池中

		String d = "ab" + "cd";
		// 如果d和c指向了同一个对象,则说明d也被加入了字符串池
		if (d == c) {
			System.out.println("\"ab\"+\"cd\" 创建的对象 \"加入了\" 字符串池中"); //答案
		}
		// 如果d和c没有指向了同一个对象,则说明d没有被加入字符串池
		else {
			System.out.println("\"ab\"+\"cd\" 创建的对象 \"没加入\" 字符串池中");
		}
		String e = a + "cd";
		// 如果e和c指向了同一个对象,则说明e也被加入了字符串池
		if (e == c) {
			System.out.println(" a  +\"cd\" 创建的对象 \"加入了\" 字符串池中");
		}
		// 如果e和c没有指向了同一个对象,则说明e没有被加入字符串池
		else {
			System.out.println(" a  +\"cd\" 创建的对象 \"没加入\" 字符串池中");//答案.
		}
		String f = "ab" + b;
		// 如果f和c指向了同一个对象,则说明f也被加入了字符串池
		if (f == c) {
			System.out.println("\"ab\"+ b   创建的对象 \"加入了\" 字符串池中");
		}
		// 如果f和c没有指向了同一个对象,则说明f没有被加入字符串池
		else {
			System.out.println("\"ab\"+ b   创建的对象 \"没加入\" 字符串池中");//答案
		}

		String g = a + b;
		// 如果g和c指向了同一个对象,则说明g也被加入了字符串池
		if (g == c) {
			System.out.println(" a  + b   创建的对象 \"加入了\" 字符串池中"); 
		}
		// 如果g和c没有指向了同一个对象,则说明g没有被加入字符串池
		else {
			System.out.println(" a  + b   创建的对象 \"没加入\" 字符串池中");//答案
		}
	}
}

 java编译器对string常量表达式的处理和优化

sun的编译器,jdk1.5是通过的如下:

 

 

首先把问题摆出来,先看这个代码

String a = "ab";
String b = "a" + "b";
System.out.println((a == b));

打印结果会是什么?类似这样的问题,有人考过我,我也拿来考过别人(蛮好玩的,大家也可以拿来问人玩),一般答案会是以下几种:

1.true
    "a" + "b" 的结果就是"ab",这样a,b都是"ab"了,内容一样所以"相等",结果true
    一般java新人如是答。
2.false
    "a" + "a"会生成新的对象"aa",但是这个对象和String a = "ab";不同,(a == b)是比较对象引用,因此不相等,结果false
    对java的String有一定了解的通常这样回答。
3.true
    String a = "ab";创建了新的对象"ab"; 再执行String b = "a" + "b";结果b="ab",这里没有创建新的对象,而是从JVM字符串常量池中获取之前已经存在的"ab"对象。因此a,b具有对同一个string对象的引用,两个引用相等,结果true.
    能回答出这个答案的,基本已经是高手了,对java中的string机制比较了解。
    很遗憾,这个答案,是不够准确的。或者说,根本没有运行时计算b = "a" + "b";这个操作.实际上运行时只有String b = "ab";
    3的观点适合解释以下情况:
    String a = "ab";
    String b = "ab";
    System.out.println((a == b));
    如果String b = "a" + "b";是在运行期执行,则3的观点是无法解释的。运行期的两个string相加,会产生新的对象的。(本文后面对此有解释)

4.true
    下面是我的回答:编译优化+ 3的处理方式 = 最后的true
    String b = "a" + "b";编译器将这个"a" + "b"作为常量表达式,在编译时进行优化,直接取结果"ab",这样这个问题退化
    String a = "ab";
    String b = "ab";
    System.out.println((a == b));
    然后根据3的解释,得到结果true

    这里有一个疑问就是String不是基本类型,像
int secondsOfDay = 24 * 60 * 60;
    这样的表达式是常量表达式,编译器在编译时直接计算容易理解,而"a" + "b" 这样的表达式,string是对象不是基本类型,编译器会把它当成常量表达式来优化吗?
    下面简单证明我的推断,首先编译这个类:
public class Test {
    private String a = "aa";
}
       复制class文件备用,然后修改为
public class Test {
    private String a = "a" + "a";
}
    再次编译,用ue之类的文本编辑器打开,察看二进制内容,可以发现,两个class文件完全一致,连一个字节都不差.
    ok,真相大白了.根本不存在运行期的处理String b = "a" + "b";这样的代码的问题,编译时就直接优化掉了。


下面进一步探讨,什么样的string + 表达式会被编译器当成常量表达式?
String b = "a" + "b";
这个String + String被正式是ok的,那么string + 基本类型呢?

String a = "a1";
String b = "a" + 1;
System.out.println((a == b));  //result = true

String a = "atrue";
String b = "a" + true;
System.out.println((a == b));  //result = true

String a = "a3.4";
String b = "a" + 3.4;
System.out.println((a == b));  //result = true
  
可见编译器对string + 基本类型是当成常量表达式直接求值来优化的。

再注意看这里的string都是"**"这样的,我们换成变量来试试:
String a = "ab";
String bb = "b";
String b = "a" + bb;
System.out.println((a == b));   //result = false
这个好理解,"a" + bb中的bb是变量,不能进行优化。这里很很好的解释了为什么3的观点不正确,如果String+String的操作是在运行时进行的,则会产生新的对象,而不是直接从jvm的string池中获取。

再修改一下,把bb作为常量变量:
String a = "ab";
final String bb = "b";
String b = "a" + bb;
System.out.println((a == b));   //result = true
竟然又是true,编译器的优化好厉害啊,呵呵,考虑下面这种情况:
String a = "ab";
final String bb = getBB();
String b = "a" + bb;
System.out.println((a == b));    //result = false
private static String getBB() {
return "b";
}
看来java(包括编译器和jvm)对string的优化,真的是到了极点了,string这个所谓的"对象",完全不可以看成一般的对象,java对string的处理近乎于基本类型,最大限度的优化了几乎能优化的地方。

另外感叹一下,string的+号处理,算是java语言里面唯一的一个"运算符重载"(接触过c++的人对这个不会陌生)吧?

 

 

网友A:

第三点其实是正确的哦, 你所说的问题在于"a"+"b"是什么时候被执行的,但是并不能解释引用a为什么是==引用b的,其实你想说的是string的+操作是在编译期执行,这个确实是你正确的,从你的例子中也可以看出来。

但是如果要解释a==b是true这个问题还是要用第3点解释哦,所以第3点并没有错,这是千真万确的,但是你对+操作的执行时期的描述也是正确的,这两个应该是一个整体,而不是对立的。

String a = "ab";
String b = "ab";
System.out.println((a == b));
问题还是那样,结果为什么是true,必须用第3点来解释,而且这也是深入浅出jvm中的观点。

 

 

直听传闻说法是:

SUN JDK 的 javac 是优化最多的, 编译速度也慢.
IBM jikes 编译速度快, 但没什么优化.
Eclipse JDT Compiler 是从 VA4J Compiler 演化来的, 增量编译很强, 优化也有, 相对JDK少.
GNU GCJ 不清楚, 不过还不成熟, 研究意义似乎不大.

不过我也没怎么接触过jikes, 楼主有空不妨用别的编译器试试.

 

 

网友M:

 

恩,就是这个意思,我的表述不是很清晰。
我所说的第三点错,并不是说第三点的解释方式有问题,而是第三点阐述的"a"+"b"是运行期被执行不正确,而是第4点钟的编译器执行优化。
第三点中阐述的jvm对string的处理我没有异议,呵呵,所以我在第4点中根本没有解释
String a = "ab";
String b = "ab";
System.out.println((a == b));
结果为什么是true,因为基本能想到第3点或第4点人,肯定非常清楚这里的true是怎么来的,呵呵。

第4点,其实是建立在第3点阐述的jvm对string的处理机制的基础上的。先执行4优化,再执行3的机制,最后才能得到true这么一个结果。

 

修改了一下文档的内容,将3的阐述改的清楚了一些。

实际这里存在两个问题
1.   3解释下面为true的问题
String a = "ab";
String b = "ab";
System.out.println((a == b));
   但是3的错误在于3中认为在运行期String b = "a"+"b";可以等同于String b = "ab";
2.   4通过编译器优化的解释解决了
String b = "a"+"b";等同于String b = "ab";的问题

 

 

还有对于jvm对string的处理和优化,我的感觉(还不确定,请大家指正)是这样,jvm里面的string池,似乎只使用于两个情况:

1.在编译后的*.class文件里面定义的"abc"这样的"直接"常量
   对于new String(), String + String/基本类型,toString()方法之类生成的string不是从string池里面取,而是直接生成新的string类。
2.调用string的intern()方法
   这个算是"强制"加入池吧

 

 

 

 

String这个东西也挺搞的,看了这个帖子,才知道我昨天做了一道题目错了。:(
立即写了个小程序测试:

Java代码 复制代码 收藏代码
	public static void main(String[] args) {
		String s1 = "ab";
		String s2="a"+"b";
		System.out.println((s1==s2)+":"+(s1.equals(s2)));
		StringBuffer sb1=new StringBuffer("ab");
		StringBuffer sb2=new StringBuffer().append("a").append("b");
		System.out.println((sb1==sb2)+":"+(sb1.equals(sb2)));
	}


结果是
true:true
false:false

对于StringBuffer,只有引用相同,==和equals()才会成立。我看了equals()源码,直接就是返回==比较的结果。
对于String,正如楼主的编译器优化的说法,我再写了一个程序:

Java代码 复制代码 收藏代码
	private static String getString(){
		return "b";
	}
	public static void main(String[] args) {
		String s1 = "ab";
		String s2="a"+getString();
		System.out.println((s1==s2)+":"+(s1.equals(s2)));
	}


结果:
false:true

这个程序运行时产生的String对象是equals的,但==不成立。

 

分享到:
评论

相关推荐

    JVM面试思考准备.zip

    在面试中,深入理解JVM对于Java开发者至关重要,因为它涉及到性能优化、内存管理、异常处理等多个关键领域。以下是对JVM面试中可能涉及的重要知识点的详细说明: 1. **JVM架构**: - **类加载子系统**:负责加载、...

    关于Java栈与堆的思考

    在Java编程语言中,理解栈(stack)与堆(heap)的概念及其工作原理对于提高程序效率、避免内存泄漏等问题至关重要。本文将深入探讨Java中的栈与堆,帮助读者更好地理解这两个核心概念,并学会如何有效利用它们。 #...

    怎样用Jvm处理Java数组.doc

    ### 如何理解Java数组在JVM中的处理方式 在探讨如何使用JVM处理Java数组之前,我们首先需要明确几个概念。Java数组本质上是一种特殊的对象,它具有固定长度,并且存储同种类型的元素。本文将围绕Java数组的一些核心...

    java基础知识思考题+答案(个人整理)

    ### Java基础知识思考题详解 #### 1. 缩写JDK的含义是什么? JDK,全称Java Development Kit,即Java开发工具包。它是Java软件开发的基础,包含了编写、编译和运行Java程序所需的所有工具。JDK不仅包括Java编译器...

    Java复习---思考题

    Java 复习资料中的思考题涵盖了Java的基础知识,包括JDK的含义、Java跨平台技术、垃圾收集机制、J2SE、J2ME、J2EE的区别、包的作用、J2SE类库的主要包以及Java编程规范和程序结构。下面将详细解答这些知识点。 1. ...

    Java string不可变原理实例解析

    在面试中,了解并能够解释Java中的String为什么是不可变的,是展示对Java内存模型和多线程概念理解的很好方式。阅读《Effective Java》等书籍可以帮助深入理解这一设计决策背后的思考。同时,熟悉字符串操作的最佳...

    java开发实习周记12篇.doc

    * JVM就是一个进程,在JVM中分出线程 * I/O:是在JVM和外部数据源之间交换数据 实习第五周 使用Collection框架做派叫号系统,学习了项目需求分析和Collection框架的应用。 这12篇文章涵盖了java开发的方方面面,从...

    Java基础-18天入门版

    首先,Java源代码通过JDK中的javac编译器被编译成字节码(.class文件),然后通过Java虚拟机(JVM)在不同的平台上解释执行字节码,最终转化为特定平台的机器码。 **Java跨平台的本质:** Java的“一次编写,到处...

    java经典面试题汇总(精华版).pdf

    在 Java 中,多态可以通过方法重载和方法覆盖来实现。 - **示例**: ```java Animal animal = new Dog(); animal.move(); // 输出 "狗可以跑和走" ``` - **作用**:增强了程序的灵活性和扩展性。 #### 三、...

    java面试题 面试必备

    在Java面试中,理解JDK与JRE的区别是非常基础且重要的。JDK(Java Development Kit)是Java开发工具包,它包含了JRE(Java Runtime Environment)以及用于开发和调试Java应用程序的各种工具,如编译器Javac、Javadoc...

    java实习周记.docx

    在第二周实习中,作者主要学习了java的基础语法,包括类之间的关系、基本数据类型、常用逻辑语句和循环语句、修饰符、Static关键字、Final关键字、This关键字、string和stringBuffer等。作者总结了学习的要点,包括...

    《Java语言程序设计(一)》课后习题答案(课程编号04747)

    - **平台独立性**:Java采用了一种名为Java虚拟机(JVM)的技术,使得Java程序可以在任何安装了JVM的操作系统上运行,实现了“一次编写,到处运行”(Write Once, Run Anywhere, WORA)的目标。 - **面向对象**:...

    Java实习周报通用25篇

    这一阶段,他们熟悉了工作环境和Java编程的环境配置,强调了“多看、多问、多观察、多思考”的学习态度。 第二周,实习生专注于Java的基础语法学习,如类之间的关系(关联、聚合、继承、实现和多态),基本数据类型...

    中南大学Java实验报告三

    Java程序的执行依赖于Java虚拟机(JVM),它使得代码能在不同的操作系统上运行,实现了“一次编写,到处运行”。 二、面向对象编程(OOP) 在Java中,面向对象编程是核心特性。实验报告可能会涵盖以下概念: 1. *...

    java学习帮助文档

    Java编程语言是软件开发领域最广泛使用的编程语言之一,尤其在企业级应用中占据主导地位。这份"java学习帮助文档"对于那些想要踏入Java世界的新手来说,无疑是一份宝贵的资源。下面,我们将深入探讨Java语言的核心...

    魔乐Java 李兴华 Java面向对象

    【Java面向对象】是编程语言Java的核心特性,它允许我们以更加抽象和模块化的方式思考问题,通过模拟现实世界中的对象来创建程序。在Java中,数据类型分为两类:基本数据类型和引用数据类型。基本数据类型包括数值型...

    Java面试指南2016

    它通过具体的技术知识点串联,模拟面试官的提问方式,引导求职者在面试过程中应该如何系统地思考问题,并用准确的技术术语表达自己的理解,从而在面试中脱颖而出。对于求职者来说,该书能够帮助他们系统地复习和掌握...

    java面试中经常会遇到的问题

    以下是一些在Java面试中经常出现的问题,包括基础、中等和难题,以及一些常见的思考题。 **基础问题:** 1. **Java是什么?** Java是一种跨平台、面向对象的编程语言,由Sun Microsystems(现为Oracle ...

    java各知识点详细总结.doc

    在编写Java代码时,遵循以下步骤有助于清晰地思考问题: 1. 明确需求:确定要实现的功能或解决问题的目标。 2. 分析思路:将需求拆分成小步骤,明确每一步的逻辑。 3. 确定步骤:将思路细化为具体的操作,涉及哪些...

Global site tag (gtag.js) - Google Analytics