String、StringBuffer、StringBulider

一、String

String:字符串,使用一对""引起来表示。

  1. String声明为final的,不可被继承

  2. String实现了Serializable接口:表示字符串是支持序列化的
    实现了Comparable接口:表示String可以比较大小

  3. String内部定义了final char[] value用于存储字符串数据

  4. String:代表不可变的字符序列,简称:不可变性

    体现:

    1. 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值。
    2. 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
    3. 当调用String.replace()方法修改指定字符或字符串时,也需要重新制定内存区域赋值,不能使用原有的value进行赋值。
  5. 通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。

  6. 字符串常量池中是不会存储相同内容的字符串的。

String的实例化方式:
方式一:通过字面量定义的方式
方式二:通过new + 构造器的方式
String s = new String(“abc”) 方式创建对象,在内存中创建了几个对象?
两个:一个是在堆空间中new结构,另一个是char[]对应的常量池中的数据:“abc”

//通过字面量定义的方式:此时的s1和s2的数据JavaEE声明在方法区中的字符串常量池中
String s1 = "JavaEE";
String s2 = "JavaEE";
//通过new + 构造器的方式:此时的s3和s4保存的地址值,是数据在堆空间中开辟空间以后对应的地址值
String s3 = new String("JavaEE");
String s4 = new String("JavaEE");

1.常量和常量的拼接结果在常量池,且常量池中不会存在相同内容的常量。
2.只要其中有一个拼接是变量,结果就在堆中。
3.如果拼接的结果调用intern()方法,返回值就在常量池中。当调用intern()方法时,如果池中已经包含一个等于此String对象的字符串,则返回池中的字符串

String s1 = "Java";
String s2 = "Python";

String s3 = "JavaPython";
String s4 = "Java" + "Python";
String s5 = s1 + "Python";
String s6 = "Java" + s2;
String s7 = s1 + s2;
System.out.println(s3 == s4); // 用 "==" 判断地址
System.out.println(s3 == s5);
System.out.println(s3 == s6);
System.out.println(s5 == s6);
System.out.println(s5 == s7);

String s8 = s5.intern();//返回值得到的s8使用的常量池中已经存在的"JavaPython"
System.out.println(s3 == s8);

二、String、StringBuffer、StringBuilder的异同

String:不可变的字符序列;底层使用char[]存储
StringBuffer:可变的字符序列;线程安全的,效率低;底层使用char[]存储
StringBuilder:可变的字符序列;jdk5.0新增的,线程不安全的,效率高;底层使用char[]存储

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

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

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

//问题一:System.out.print(sb2.length());//3
//问题二:扩容问题;如果要添加的数据底层数组盛不下了,那就需要扩容底层的数组。
newCapacity();StringBuffer、StringBuilder 都继承了 AbstractStringBuilder,其中有一个 newCapacity() 方法,默认情况下,扩容为原来容量的2倍 + 2,同时将原有数组中的元素复制到新的数组中。

//开发中建议大家使用:StringBuffer(int capacity) 或 StringBuilder(int capacity)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值