Java String对象的经典问题
先来看一个样例,代码例如以下:
- public class Test {
- public static void main(String[] args) {
- String str = "abc";
- String str1 = "abc";
- String str2 = new String("abc");
- System.out.println(str == str1);
- System.out.println(str1 == "abc");
- System.out.println(str2 == "abc");
- System.out.println(str1 == str2);
- System.out.println(str1.equals(str2));
- System.out.println(str1 == str2.intern());
- System.out.println(str2 == str2.intern());
- System.out.println(str1.hashCode() == str2.hashCode());
- }
- }
假设您能对这8个输出结果直接推断出来。以下的分析就不用看了。可是我想还是有非常多人对这个String对象这个问题仅仅是表面的理解,以下就来分析一下Java语言String类和对象及其执行机制的问题。
做个基础的说明。堆(heap)内存和栈(Stack)内存的问题。
堆和栈的数据结构这里就不解释了。Java语言使用内存的时候,栈内存主要保存以下内容:基本数据类型和对象的引用,而堆内存存储对象,栈内存的速度要快于堆内存。
总结成一句话就是:引用在栈而对象在堆。
Java中的比較有两种,是==和equals()方法,equals()是Object类的方法,定义在Object类中的equals()方法是例如以下实现的:
- public boolean equals(Object obj){
- return (this==obj);
- }
String类重写了equals()方法,改变了这些类型对象相等的原则,即推断对象是否相等根据的原则为推断二者的内容是否相等。
了解以上内容后我们来说说String,String类的本质是字符数组char[],其次String类是final的。是不可被继承的,这点可能被大多数人忽略,再次String是特殊的封装类型。使用String时能够直接赋值。也能够用new来创建对象,可是这二者的实现机制是不同的。另一个String池的概念,Java执行时维护一个String池。池中的String对象不可反复,没有创建,有则作罢。String池不属于堆和栈。而是属于常量池。以下分析上方代码的真正含义
- String str = "abc";
- String str1= "abc";
第一句的真正含义是在String池中创建一个对象”abc”,然后引用时str指向池中的对象”abc”。第二句执行时,由于”abc”已经存在于String池了,所以不再创建,则str==str1返回true就明确了。str1==”abc”肯定正确了,在String池中仅仅有一个”abc”。而str和str1都指向池中的”abc”,就是这个道理。
- String str2 = new String("abc");
这个是Java SE的热点问题,众所周知,单独这句话创建了2个String对象,而基于上面两句,仅仅在栈内存创建str2引用。在堆内存上创建一个String对象,内容是”abc”。而str2指向堆内存对象的首地址。
以下就是str2==”abc”的问题了,显然不正确,”abc”是位于String池中的对象,而str2指向的是堆内存的String对象。==推断的是地址,肯定不等了。
str1.equals(str2)。这个是对的。前面说过,String类的equals重写了Object类的equals()方法,实际就是推断内容是否同样了。
以下说下intern()方法,在JavaDoc文档中。这样描写叙述了intern()方法:返回字符串对象的规范化表示形式。
怎么理解这句话?实际上过程是这样进行的:该方法如今String池中查找是否存在一个对象,存在了就返回String池中对象的引用。
那么本例中String池存在”abc”,则调用intern()方法时返回的是池中”abc”对象引用,那么和str/str1都是等同的,和str2就不同了,由于str2指向的是堆内存。
hashCode()方法是返回字符串内容的哈希码,既然内容同样。哈希码必定同样,那他们就相等了,这个easy理解。
再看以下的样例:
- public class Test {
- private static String str = "abc";
- public static void main(String[] args) {
- String str1 = "a";
- String str2 = "bc";
- String combo = str1 + str2;
- System.out.println(str == combo);
- System.out.println(str == combo.intern());
- }
- }
这个样例用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,显然str==combo是错误的,而str==combo.intern()是正确的。在String池中也存在”abc”。那就直接返回了,而str也是指向String池中的”abc”对象的。
此例说明不论什么又一次改动String都是又一次分配内存空间,这就使得String对象之间互不干扰。也就是String中的内容一旦生成不可改变。直至生成新的对象。
同一时候问题也来了。使用+连接字符串每次都生成新的对象,并且是在堆内存上进行,而堆内存速度比較慢(相对而言)。那么再大量连接字符串时直接+是不可取的。当然须要一种效率高的方法。Java提供的StringBuffer和StringBuilder就是解决问题的。
差别是前者是线程安全的而后者是非线程安全的,StringBuilder在JDK1.5之后才有。
不保证安全的StringBuilder有比StringBuffer更高的效率。
自JDK1.5之后。Java虚拟机执行字符串的+操作时。内部实现也是StringBuilder,之前採用StringBuffer实现。
关于 在Javak 连接字符串时是使用+号不审StringBuilder。可參考http://blog.csdn.net/atomic_age/article/details/1656964
欢迎交流,希望对使用者实用。
Java String对象的经典问题的更多相关文章
- Java String对象的经典问题(转)
public class StringTest { public static void main(String[] args) { String strA = "abc"; St ...
- JAVA String对象和字符串常量的关系解析
JAVA String对象和字符串常量的关系解析 1 字符串内部列表 JAVA中所有的对象都存放在堆里面,包括String对象.字符串常量保存在JAVA的.class文件的常量池中,在编译期就确定好了 ...
- 深入理解java String 对象的不可变性
下面我们通过一组图表来解释Java字符串的不可变性 1.声明一个String对象 String s = "abcd"; 2.将一个String变量赋值给另一个String变量 St ...
- 关于Java String对象创建的几点疑问
我们通过JDK源码会知道String实质是字符数组,而且是不可被继承(final)和具有不可变性(immutable).可以如果想要了解String的创建我们需要先了解下JVM的内存结构. 1.JVM ...
- Java——String对象
前言 实际上任何语言都没有提供字符串这个概念,而是使用字符数组来描述字符串.Java里面严格来说也是没有字符串的,在所有的开发里面字符串的应用有很多,于是Java为了应对便创建了String类这个字符 ...
- Java String对象的问题 String s="a"+"b"+"c"+"d"
1, String s="a"+"b"+"c"+"d"创建了几个对象(假设之前串池是空的) 2,StringBuilde ...
- 我的Java开发学习之旅------>Java String对象作为参数传递的问题解惑
又是一道面试题,来测试你的Java基础是否牢固. 题目:以下代码的运行结果是? public class TestValue { public static void test(String str) ...
- Java String 对象,你真的了解了吗?
String 对象的实现 String对象是 Java 中使用最频繁的对象之一,所以 Java 公司也在不断的对String对象的实现进行优化,以便提升String对象的性能,看下面这张图,一起了解一 ...
- java String对象的创建(jvm).
本人目前也开始学习虚拟机,在java中,有很多种类型的虚拟机,其中就以sum公司(当然现在已经是oracle了)的虚拟机为例,介绍可能在面试的时候用到的,同时对自己了解String有很大帮助,这里仅仅 ...
随机推荐
- caioj1441:第k小的数Ⅰ
[传送门:caioj1441] 简要题意: 给出一个n个数的序列,m个询问,每个询问输入l,r,k,输出第l个数到第r个数第k小的数 题解: 首先想到线段树,但是做不到询问区间的第几小,只能做到最大或 ...
- Uncaught DOMException: Failed to execute 'toDataURL' on 'HTMLCanvasElement': Tainted canvases may not be exported
Uncaught DOMException: Failed to execute 'toDataURL' on 'HTMLCanvasElement': Tainted canvases may no ...
- route---设置Linux内核中的网络路由表
route命令用来显示并设置Linux内核中的网络路由表,route命令设置的路由主要是静态路由.要实现两个不同的子网之间的通信,需要一台连接两个网络的路由器,或者同时位于两个网络的网关来实现. 在L ...
- 【Henu ACM Round#14 C】Duff and Weight Lifting
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 2^y可以由两个2^(y-1)相加得到. 则有一个贪心的策略. 就是2^x尽量都变成2^(x+1) (即能够凑就尽量凑) 如果x还有 ...
- CODEVS——T 1404 字符串匹配
http://codevs.cn/problem/1404/ 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 大师 Master 题解 查看运行结果 题目描述 Desc ...
- PIM-DM协议内核触发机制及协议执行机制记录
PIM-DM和PIM-SM是组播路由ASM(随意信源组播)中的两种不同模式.相对PIM-SM的组播注冊机制.PIM-DM的扩散机制显得更加粗犷. 一.PIM-DM无需向内核注冊pimreg虚接口. 可 ...
- SQL Server 函数的使用 Function
create table student ( id varchar2(5) primary key, name varchar2(20) not null, sex char(2) check(sex ...
- BZOJ 2708 [Violet 1]木偶 DP
题意:id=2708">链接 方法: DP 解析: 这题太神辣. 做梦都没想到DP啊,反正我不会. 先谈一个我有过的错的想法. 最小费用最大流? 能匹配的边连费用为1的,不能匹配的连费 ...
- POJ 2185 正解 KMP
题意: 思路: 把每一行压成一个数 求一下 KMP 把每一列压成一个数 求一下KMP 答案就是两个周期之积 网上的好多题解都是错的---------.. //By SiriusRen #include ...
- SuSe Linux Enterprise Server 10 With Sp2 安装过程图解
SuSe Linux Enterprise Server 10 With Sp2 安装过程图解 650) this.width=650;" style="border-right- ...