从概念上讲,Java中字符串就是Unicode字符序列。例如,"Hello"由H、e、l、l、o 5个字符组成,Java没有内置的字符串类型,而是在标准Java类库中提供了一个预定义类,很自然的叫做String。每个双引号括起来的字符串都是String类的一个实例。
一、创建String类
String类型对象主要有两种创建方式,第一种字面量赋值:String h = "hello";最常用的方式,如此方式,先查询字符串常量池,equlas判断字符串常量池中有无为true的对象引用。没有则在堆内存中创建一个"hello"对象本体,并在字符串常量池中添加,刚创建的堆内存中"hello"对象引用。于此同时,如果再声明一个String对象,String e = "hello";则不会再创建一个String对象,而是和变量h一样获得字符串常量区的该String对象的引用。
第二种,String h = new String("hello");这种方式创建的String对象,既有可能创建出两个对象,也可能创建出一个对象。如果此时字符串常量区没有值为"hello"对象,则会创建一个引用对象(1.6是永久复制一个String对象,1.7后改为复制实例的引用),如果有则只创建一个在堆内存中(String对象本体);
String对象的值是不可变的,重新复制String变量只是改变其引用,原对象没有改变(1.8之前是char数组,1.8以后改为byte数组)
二、== 和 equlas()
先回顾一下,==和equls()的区别,==是比较二者值是否相等,用于基本类型是直接比较二者的值是否相等,用于对象则是比较二者的引用内存地址是否相等。
那么在很多时候,我们需要对String类型对象进行比较,尤其是面试时经常容易问到相关字符串==比较的问题,所以我们必须搞懂它。
面试中常考的例子是:
a变量赋值对象创建时,因为没有在字符串常量区里找到equals为true的String。所以在堆中创建一个"abc"对象,并且在非堆字符串常量池中创建对在堆中刚创建abc对象的引用,注意是引用,不是对象实体,然后把此引用赋给a变量。
b变量赋值对象创建时,因为是new了一个String对象,虽然值还是"abc",但是却是在堆中新创建了一个对象,与上边的不一样,有不同的内存地址。
a变量赋值对象创建后,堆中会创建一个"abc"对象,然后在非堆字符串常量池中创建其引用,然后b和c因为equlas到了相同对象,所以不创建新的对象,直接从字符串常量池中获得"abc"的引用,因此==判断时,此三个变量的值都相等,因为指向的是同一个对象。
还没清楚的话,建议看这两篇文章:
java用这样的方式生成字符串:String str = "Hello",到底有没有在堆中创建对象? - 胖君的回答 - 知乎
请别再拿“String s = new String("xyz");创建了多少个String实例”来面试了吧
String的equals方法,会比对两个对象是否是一个对象,如果不是那么就逐一的检查其byte数组其每一个数值是否相等,都相等返回true。
String类型的空串表示为"",长度为0;空串为null,因为String对象的变量也是引用类型,所以可以为其赋null值。因此判断String类型有没有值时,需要检查 是否为null和是否长度为0。
三、String的intern()方法
intern方法,从1.7开始,如果常量池中已经有了这个字符串,那么直接返回常量池中的它的引用,如果没有就将该字符串的对象的引用保存到常量池中并返回。
知道了intern的功能了,那么它有什么用处?
在上边的例子中,第一行第二行并没有创建"hello"字符串的实例,而第三行在常量池中创建了"hello"的引用,因此c、d拿到了同一个对象的引用,所以相等。
四、String类的常用API
只列举几个常用的方法,全部的可以查看官方文档或String类源码。
1、int length() 获取字符串的长度,基本上是最常用的方法
2、String trim() 获取去除空格后的字符串
3、String[] split(String regex,int limit) 根据regex字符串对象值匹配分割字符串生成数组,limit代表分割后数组最大长度
4、String[] split(String regex) 根据regex字符串对象值匹配分割字符串生成数组
5、String[] substring(int beginIndex) 从beginIndex索引位置开始截取,去掉beginIndex之前的部分。
6、String[] substring(int beginIndex,int endIndex) 从beginIndex索引位置开始截取到endIndex(不包括endIndex),去掉beginIndex之前的部分和endIndex与endIndex之后的部分。
7、boolean equals(Object anObject) 传入对象与当前String对象为一个对象,或者值相等时返回true,反之false