全目标70分之 Java进阶(六):String类的使用和部分源码分析

    从概念上讲,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数组)


final修饰,无法改变其值

二、== 和 equlas()

    先回顾一下,==和equls()的区别,==是比较二者值是否相等,用于基本类型是直接比较二者的值是否相等,用于对象则是比较二者的引用内存地址是否相等。

    那么在很多时候,我们需要对String类型对象进行比较,尤其是面试时经常容易问到相关字符串==比较的问题,所以我们必须搞懂它。

    面试中常考的例子是:


输出结果是false,因为是两个不同的对象

    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。


第一个if先判断==,判断二者引用是否相同,如果不是,然后根据编码格式比较值
对比二者的byte数组,按顺序检查是否每一个元素的值都相等。

        String类型的空串表示为"",长度为0;空串为null,因为String对象的变量也是引用类型,所以可以为其赋null值。因此判断String类型有没有值时,需要检查 是否为null和是否长度为0。

三、String的intern()方法

    intern方法,从1.7开始,如果常量池中已经有了这个字符串,那么直接返回常量池中的它的引用,如果没有就将该字符串的对象的引用保存到常量池中并返回。

    知道了intern的功能了,那么它有什么用处?


前两行并没有创建"hello"对象,第三行调用intern方法向常量池中添加了一个引用

    在上边的例子中,第一行第二行并没有创建"hello"字符串的实例,而第三行在常量池中创建了"hello"的引用,因此c、d拿到了同一个对象的引用,所以相等。


可见第一行代码运行后,常量池中并没有"hello"的引用,第二行时才添加,所以c和d==e

四、String类的常用API

    只列举几个常用的方法,全部的可以查看官方文档或String类源码。

    1、int length()  获取字符串的长度,基本上是最常用的方法

    2、String trim() 获取去除空格后的字符串

    3、String[] split(String regex,int limit) 根据regex字符串对象值匹配分割字符串生成数组,limit代表分割后数组最大长度


也可以理解把字符串最大分割为limit-1个字符串

    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之后的部分。

两个substring方法的区别

       7、boolean equals(Object anObject) 传入对象与当前String对象为一个对象,或者值相等时返回true,反之false

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。