java中String、StringBuffer、StringBuilder

一、String类

概述

String类是final类,不能被继承,底层用的数据结构是数组
通过字面量的方式进行赋值,java中 仅此一份。
特性
1.String声明为final的,不可被继承
2.String实现了Serializable接口:表示字符串是支持序列化的。
实现了Comparable接口:表示String可以比较大小
(我们熟知的还有什么接口啊:comparator接口)
3.String内部定义了final char[] value用于存储字符串数据
4.String:代表不可变的字符序列。简称:不可变性。
体现:1.当对字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值。
2. 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
3. 当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
5.通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。
6.字符串常量池中是不会存储相同内容的字符串的。
7. String类重写了Object中的equals方法,依次比较的顺序是地址->长度->每个字符

String的常用方法

length()/charAt()/equals()/compareTo()/contains()/indexOf()/getBytes()/tocharArray()/valueOf()/matches(String regex)/

String使用陷阱
  • String s1 = “a”;
    说明:在字符串常量池中创建了一个字面量为"a"的字符串。
  • s1 = s1 + “b”;
    说明:实际上原来的“a”字符串对象已经丢弃了,现在在堆空间中产生了一个字符
    串s1+“b”(也就是"ab")。如果多次执行这些改变串内容的操作,会导致大量副本
    字符串对象存留在内存中,降低效率。如果这样的操作放到循环中,会极大影响
    程序的性能。
  • String s2 = “ab”;
    说明:直接在字符串常量池中创建一个字面量为"ab"的字符串。
  • String s3 = “a” + “b”;
    说明:s3指向字符串常量池中已经创建的"ab"的字符串。
  • String s4 = s1.intern();
    说明:堆空间的s1对象在调用intern()之后,会将常量池中已经存在的"ab"字符串
    赋值给s4
  • 常量与常量的拼接结果在常量池。且常量池中不会存在相同内容的常量。(这里注意将变量加上final修饰后就变成常量了)
  • 只要其中有一个是变量,结果就在堆中
  • 如果拼接的结果调用intern()方法,返回值就在常量池中
String的值传递

涉及String的值传递
基本数据类型传递的是数据
引用数据类型传递的是地址值
这里并不是说引用数据类型是不可变的,只是引用数据如果是String,会比较特殊。

public class StringTest {

    String str = new String("good");
    char[] ch = { 't', 'e', 's', 't' };

    public void change(String str, char ch[]) {
        str = "test ok";//你改了,并不影响别人
        ch[0] = 'b';
    }
    public static void main(String[] args) {
        StringTest ex = new StringTest();
        ex.change(ex.str, ex.ch);
        System.out.println(ex.str);//good
        System.out.println(ex.ch);//best
    }
}

当上面的ex.str地址传递给str后,str="test ok‘这句代码会在常量池中声明”test ok“这个字符串,但是本身ex.str在堆中的指向依然是"good"这个字符串。但是下面的char字符串就不一样。

JVM 将堆空间划分成三部分(栈、堆、方法区),其中将字符串常量池放在方法区中。
我们需要注意:数组也是存放在堆内存中的,数组在java中不属于基本类型,而是属于引用类型的成员变量。
一般来说通过普通的传值方式写的代码都是属于浅拷贝的范畴,对于浅拷贝而言:
浅拷贝(Shallow Copy)
对于基本数据类型的成员变量,浅拷贝进行的是值传递,也就是将对象A中的基本数据类型的成员变量的值赋值给对象B中的同等基本数据类型的成员变量。因为是两份不同的数据,所以对其中一个对象的该成员变量值进行修改,不会影响另一个对象拷贝得到的数据
对于数据类型是引用类型的成员变量,比如说成员变量是某个数组、某个类的对象等,那么浅拷贝会进行引用传递,也就是只是将该成员变量的引用值(内存地址)复制一份给新的对象。因为实际上两个对象的该成员变量都指向同一个实例。在这种情况下,在一个对象中修改该成员变量会影响到另一个对象的该成员变量值
与浅拷贝相对应的就是深拷贝
深拷贝
深拷贝对引用数据类型的成员变量的对象图中所有的对象都开辟了内存空间;而浅拷贝只是传递地址指向,新的对象并没有对引用数据类型创建新的内存空间

实现深浅拷贝都可以通过实现Cloneable接口来完成,并且重写其中的**clone()**方法来完成。,只不过深拷贝中需要注意对于拷贝对象的操作
深拷贝还可以通过实现序列化接口来实现。

String类与其他结构之间的转换

注意:能够实现强制类型转换的必须是具有子父类继承关系的才可以进行强转。

  1. String—>基本数据类型、包装类
    调用包装类的静态方法:parseXxx(str)
  2. 基本数据类型、包装类—>String
    调用String重载的valueOf(xxx),记住valueOf不是静态的,故需要对象调用,不能直接类调用。
  3. 字符数组转化为字符串
  4. 字符串转化为字符数组
  5. 字符串转化为字节数组
  6. 字节数组转化为字符串
  7. 字符串转化为字节数组(注意编码方式不同带来的乱码问题)

二、StringBuffer、StringBuilder

常用方法:
增append

改setCharAt(),repalce()
查charAt()
插insert()
长度、
StringBuffer、StringBuilder都有reverse()方法,String是没有该方法的。
substring(int begin,int end)是需要接受返回值的,不像其他方法直接对当前str进行切割。

三者的异同

String是不可变的字符序列,底层实 现使用char[]数组
StringBuffer:可变的字符序列,线程安全,效率低,底层实现使用char[]数组
StringBuilder:可变的字符序列,线程不安全,效率高,底层实现使用char[]数组

  源码分析:
    String str = new String();//char[] value = new char[0];
    String str1 = new String("abc");//char[] value = new char[]{'a','b','c'};

   StringBuffer sb1 = new StringBuffer();//char[] value = new char[16];底层创建了一个长度是16的数组。
    System.out.println(sb1.length());//
    sb1.append('a');//value[0] = 'a';
    sb1.append('b');//value[1] = 'b';

   StringBuffer sb2 = new StringBuffer("abc");//char[] value = new char["abc".length() + 16];

   //问题1. System.out.println(sb2.length());//3
    //问题2. 扩容问题:如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。
      默认情况下,扩容为原来容量的2+ 2,同时将原有数组中的元素复制到新的数组中。
指导意义:开发中建议大家使用:StringBuffer(int capacity)StringBuilder(int capacity),防止频繁的扩容

对比String、StringBuffer、StringBuilder三者的效率:
从高到低排列:StringBuilder > StringBuffer > String

null值带来的思考

分析一段代码

String str = null;
StringBuffer sb = new StringBuffer();
sb.append(str);
System.out.println(sb.length());//4
System.out.println(sb);//"null"双引号中的null
StringBuffer sb1 = new StringBuffer(str);//抛异常nullPointerException
System.out.println(sb1);//

需要点击源码去看,StringBuffer的构造器到底会做什么,他会直接执行str.length(),而这时候str是null

一个别人的博客:https://baijiahao.baidu.com/s?id=1700540533913549064&wfr=spider&for=pc

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值