一、String
String:字符串,使用一对””引起来表示。
-
String声明为final的,不可被继承
-
String实现了Serializable接口:表示字符串是支持序列化的
实现了Comparable接口:表示String可以比较大小 -
String内部定义了final char[] value用于存储字符串数据
-
String:代表不可变的字符序列,简称:不可变性
体现:
- 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值。
- 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
- 当调用String.replace()方法修改指定字符或字符串时,也需要重新制定内存区域赋值,不能使用原有的value进行赋值。
-
通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。
-
字符串常量池中是不会存储相同内容的字符串的。
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)