这是对于String类的一些总结,我将会从几个方面并且结合着字符串池等相关知识进行总结
进程如下: 1.对于String类基本知识的一些总结
2.简要介绍字符串池
3.分析字符串池在内存中所起到的作用以及其在内存中的使用方式
4.对于两种实例化对象方式的不同进行描述
开始:
1.对于String类基本知识的一些总结。
首先我们清楚String类对象的值一经确认便是无法改变的 具体原因如下(我们通过查看String源码的方式来说明问题)
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
private final char value[]; // 请注意此处,String类对象是以数组的形式进行存储的,value数组便是其存储场所而该数组是以final修饰的
所以属于常量无法被改变
private int hash; private static final long serialVersionUID = -6849794470754667710L; private static final ObjectStreamField[] serialPersistentFields =
new ObjectStreamField[0];
}
其次字符串是引用性数据类型,并不属于基本数据类型(老生常谈了)
2.简要介绍字符串池(String pool)
在工作中,String类是我们使用频率非常高的一种对象类型。JVM为了提升性能和减少内存开销,避免字符串的重复创建,其维护了一块特殊的内存空间。字符串池由String类私有的维护。
3.分析字符串池在内存中所起的作用以及其作用方式
上文提到了,字符串池是一块特殊的内存空间,并且他是存在DataSegument中的(这里我也是现查的,对于DataSegument也并不是很理解),而我们所·实例化的对象都是存在于堆内存中,那么他们之间的关系又如何那?
其实我们都知道要给一个String类对象赋值我们有两种方式,一种是通过直接赋值即:
String a="顶天立地";
另一种是以new对象的方式赋值: String a=new String("哈哈哈");
对于这两种方式具体的区别,我们在下面会详细解释的,这里引用他们只是为了说明无论是何种方式,只要我们想给对象以字符串的形式赋值,那么就必须通过字符串池的确定才可以,如果字符串池中有的话,没问题可以继续往下操作了(下文会详细描述下面的操作是啥滴。。。。);如果没有的话,则会在字符串池中创建一个该字符串并且将地址给我们所要实例化的那个对象,当然啦,有的话也会给地址的。
4.通过两种实例化方式的不同进行比较
在此我会通过比较,画图等方式来加深理解
下面我会用到equals方法,我们应该知道String类中的equals方法是经过重写的所以他比较的 不再是两个对象的地址 而是字符串内容是否相等
首先我们对直接赋值的方式进行分析
String s="haha";
String a="haha";
System.out.print(s.equals(a));
System.out.println(a==s);
返回值为 true true
分析下其过程吧 首先“哈哈”(我们应该知道一个字符串也是相当于一个对象的)进入字符串池字符串池发现他没有所以新建了一个并且把“哈哈”在内存中的地址交给了s
所以说s存的其实是地址 然后当另一个“哈哈”进入字符串池时,因为已经存在了一个“哈哈”,所以字符串池本着节省资源的原则索性把上一个“哈哈”的地址给了a
所以他们不仅内容一样,而且地址还一样,当然这里并不是那么的难以理解所以就不画图了
其次我们对通过new的方式赋值的情况进行分析:
String s=new String("haha");
String a=new String("haha");
System.out.print(s.equals(a));
System.out.println(a==s);
返回结果: true false
第一个结果我想我们基本都可以预料到,但是为啥第二个会是false那?
上面我们已经了解过了,a==s相当于是比较地址了,那么地址是在哪一步发生了改变那?
对 还有一个点: 我们以new的方式进行赋值相当于实例化了两个对象
为什么这样说 我们以前也学过有new必有对象的生成,那么上文也说了“哈哈”这种字符串也是相当于一个对象的。
好,下面通过画图的方式来说明
从图中可以看出是我们所new的对象地址不同导致结果为false。