- 浏览: 7246142 次
- 性别:
- 来自: 上海
文章分类
- 全部博客 (1546)
- 企业中间件 (236)
- 企业应用面临的问题 (236)
- 小布Oracle学习笔记汇总 (36)
- Spring 开发应用 (54)
- IBatis开发应用 (16)
- Oracle基础学习 (23)
- struts2.0 (41)
- JVM&ClassLoader&GC (16)
- JQuery的开发应用 (17)
- WebService的开发应用 (21)
- Java&Socket (44)
- 开源组件的应用 (254)
- 常用Javascript的开发应用 (28)
- J2EE开发技术指南 (163)
- EJB3开发应用 (11)
- GIS&Mobile&MAP (36)
- SWT-GEF-RCP (52)
- 算法&数据结构 (6)
- Apache开源组件研究 (62)
- Hibernate 学习应用 (57)
- java并发编程 (59)
- MySQL&Mongodb&MS/SQL (15)
- Oracle数据库实验室 (55)
- 搜索引擎的开发应用 (34)
- 软件工程师笔试经典 (14)
- 其他杂项 (10)
- AndroidPn& MQTT&C2DM&推技术 (29)
- ActiveMQ学习和研究 (38)
- Google技术应用开发和API分析 (11)
- flex的学习总结 (59)
- 项目中一点总结 (20)
- java疑惑 java面向对象编程 (28)
- Android 开发学习 (133)
- linux和UNIX的总结 (37)
- Titanium学习总结 (20)
- JQueryMobile学习总结 (34)
- Phonegap学习总结 (32)
- HTML5学习总结 (41)
- JeeCMS研究和理解分析 (9)
最新评论
-
lgh1992314:
[u][i][b][flash=200,200][url][i ...
看看mybatis 源代码 -
尼古拉斯.fwp:
图片根本就不出来好吧。。。。。。
Android文件图片上传的详细讲解(一)HTTP multipart/form-data 上传报文格式实现手机端上传 -
ln94223:
第一个应该用排它网关吧 怎么是并行网关, 并行网关是所有exe ...
工作流Activiti的学习总结(八)Activiti自动执行的应用 -
ZY199266:
获取不到任何消息信息,请问这是什么原因呢?
ActiveMQ 通过JMX监控Connection,Queue,Topic的信息 -
xiaoyao霄:
DestinationSourceMonitor 报错 应该导 ...
ActiveMQ 通过JMX监控Connection,Queue,Topic的信息
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这个东西也挺搞的,看了这个帖子,才知道我昨天做了一道题目错了。:(
立即写了个小程序测试:
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,正如楼主的编译器优化的说法,我再写了一个程序:
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的,但==不成立。
发表评论
-
java中 避免过多的使用同步
2012-11-22 15:41 2201避免过多的使用同步 ... -
Java Thread的一点知识(III)
2011-10-28 15:32 3320同步和异步有何异同,阻塞与非阻塞,在什么情况下分别使用他们?举 ... -
Java Thread的一点知识(II)
2011-10-28 15:30 5388stop() 和 suspend() 方法为何不推荐使用? ... -
Java Thread的一点知识
2011-10-28 15:24 3665线程与进程的区别 线程和进程是不同的,每个进程 ... -
【转】Java内部类
2011-10-28 13:18 1526简单的说,内部(inner)类指那些类定义代码被置于其它类定义 ... -
【转】华为的JAVA面试题
2011-10-28 09:20 2318对我个人来说,看看 ... -
java Final的复习
2011-10-24 16:54 1546package com; /** *如果fina ... -
【转】 不用中间变量,实现两个变量的交换
2010-11-27 17:30 2840常见的面试题,很简单,但值得细想的问题还是很多,总结解法如下: ... -
对象池技术基本原理和分析应用(二)
2010-07-02 21:27 1933通用对象池的实现 对象池的构造和管理可以按照多种方式 ... -
对象池技术基本原理和分析应用(一)
2010-07-02 21:25 2016本文在分析对象池技 ... -
java回调机制的使用
2010-06-03 17:09 1916学习过Hibernate和Spring等童鞋,经常见到的 ... -
java 数组知道少?
2010-05-19 17:45 2314package com.easyway.common ... -
面向对象程序设计(OOP)的六大原则
2010-05-15 20:54 3015面向对象程序设计(OOP) ... -
Java 实例内部类的学习
2010-05-15 20:47 1732package com.easyway.commons.isp ... -
Java 匿名类的学习应用
2010-05-15 20:39 2033java 接口的匿名内部类 package com.ea ... -
java 异常的特殊的典型使用你知道吗?
2010-05-15 20:36 1558package com.easyway.commons.isp ... -
java 字符串类型知道少
2010-05-15 20:33 1730package com.easyway.commons.isp ... -
关于java 类的生命周期 深入了解
2010-05-15 20:15 2317package com.easyway.commons.isp ... -
枚举类的使用特点总结
2010-05-11 09:20 2328package com.easyway.commons.isp ... -
关于Integer的深度分析以及注意点
2010-05-10 22:20 2060在Integer中valueOf()和创建独享In ...
相关推荐
Java JVMJava JVMJava JVM
java 查看JVM中所有的线程的活动状况 java 查看JVM中所有的线程的活动状况
java对象在jvm中的存储情况 jvm
java11-jvm白皮书
Java Jar JVM Exe 可执行文件 Java Jar JVM Exe 可执行文件 Java Jar JVM Exe 可执行文件 Java Jar JVM Exe 可执行文件 Java Jar JVM Exe 可执行文件
2017年最新的关于jvm的书,里面还包含了如何编译openjdk部分,值得一看。
《实战Java虚拟机——JVM故障诊断与性能优化》内容简介:随着越来越多的第三方语言(Groovy、Scala、JRuby等)在Java虚拟机上运行,Java也俨然成为一个充满活力的生态圈。本书将通过200余示例详细介绍Java虚拟机中的...
java基础之JVM,jvm是java的核心技术,也是面试过程中经常出现的
主要介绍了Java对象在JVM中的生命周期详解,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
对JVM规范的的抽象说明是一些概念的集合,它们已经在书《The Java Virtual Machine Specification》(《Java虚拟机规范》)中被详细地描述了;对JVM的具体实现要么是软件,要么是软件和硬件的组合,它已经被、
实战Java虚拟机——JVM故障诊断与性能优化.pdf 实战Java虚拟机——JVM故障诊断与性能优化.pdf
Java助力需要jvm学习及优化与性能瓶颈分析参考
实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送) 实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送) 实战JAVA虚拟机 JVM故障诊断与性能优化.pdf (无书签.低分放送)
实战Java虚拟机_JVM故障诊断与性能优化 JVM故障诊断与性能优化 JVM调优
NeatJVM,java中的jvm实现.zip
Java虚拟机(JVM)面试题 51道.pdf
java版本的<write your own java virtual machine>。
JAVA应用JVM原理及参数调优深入讲解视频.1 JAVA应用JVM原理及参数调优深入讲解视频.2 JAVA应用JVM原理及参数调优深入讲解视频.3 JAVA应用JVM原理及参数调优深入讲解视频.4 JAVA应用JVM原理及参数调优深入讲解视频.5 ...
java 11相关的jvm文档 帮助了解java11的新知识
java获得jvm内存大小