String、StringBuffer、StringBulider

  • Post author:
  • Post category:其他




一、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)



版权声明:本文为weixin_43197640原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。