在讨论堆与栈,对象与引用时,我发现了String在这个怪胎。为什么说它是怪胎呢?因为它同时具备基本数据类型与非基本数据类型的特点,下面我举出例子:
两类赋值方式
- 按照对象原本方式:
- 无参构造
a="abc";```
* 有参构造
```String a =new String(“abc”);```
* 按照主数据类型方式
```String a = "abc"; //作为一个对象,你让我如何看待你```
************************
#### “==”那些坑
* 无参构造
```String b =new String();
b="abc";
System.out.print(a==b);
结果:true```
* 有参构造
```String b =new String(“abc”);
System.out.print(a==b);
结果:flase```
* 主数据类型方式
```String b = "abc";
System.out.print(a==b);
结果:true```
这到底是怎么回事呢?有兴趣去看看String是怎么包装的吧,下面进入String Pool正题。
***************
#### 字符串池
原来由于String超级常用,JVM为了提升性能和减少内存开销,避免字符串的重复创建,其维护了一块特殊的内存空间,即字符串池(String Pool)。字符串池由String类私有的维护。
我们可以根据这一特性将三种方法再分成两类:
* 无参构造与直接创建(假设均为第一次创建)
在执行a="abc";时先在字符串池中查找是否存在字符串"abc",若没有则将其存储进字符串池,同时使a的引用指向"abc",再次执行b="abc";时,重复前面操作,存在直接将b的引用指向"abc",不需要再创造新值。
* 有参构造
在调用其构造函数时便已经对引用赋值,使其跳过查找的过程,同时分配a,b两个不同指向空间,并存入字符串。
这里自然能知道两者区别;一个省内存,但是有遍历过程;一个省时间,但内存开销较前者大。不过相比之下,好像推荐前者哦。
**************
#### 那些变种
```1 String str ="ABCDEFG";
2 String strone ="AB";
3 String strtwo ="CDEFG";
4 System.out.print(str=="AB"+"CDEFG");
5 System.out.print(str==strone+strtwo);
结果:ture,false```
因为第4行在进行编译时后项先组装为"ABCDEFG",遍历能得到值;而变量是一个个对应查找的,地址对不上哦!
****************
包内外同值字符串是否相等问题,请参考:
[Java字符串池(String Pool)深度解析](http://www.cnblogs.com/fangfuhai/p/5500065.html)
[Java中String直接赋值和使用new的区别](http://blog.csdn.net/OREO_GO/article/details/51397903)