String 特性
String:字符串,使用一对" "引起来表示
特性:
-
String声明为final的,不可被继承
-
String实现了Serializable接口:表示字符串是支持序列化的。
实现了Comparable接口:表示String可以比较大小 -
String内部定义了final char[] value用于存储字符串数据
-
String:代表不可变的字符序列。简称:不可变性。
体现:
1.当对字符串重新赋值时,需要重写指定内存区域赋值,不能使用原有的value进行赋值。
2. 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
3. 当调用String的replace()方法修改指定字符或字符串时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值。
- 通过字面量的方式(区别于new)给一个字符串赋值,此时的字符串值声明在字符串常量池中。
- 符串常量池中是不会存储相同内容的字符串的
String的实例化方式:
方式一:通过字面量定义的方式
String s1 = “javaEE”;
- 此时的s1的数据javaEE声明在方法区中的字符串常量池中。
方式二:通过new + 构造器的方式
String s3 = new String(“javaEE”);
- 此时的s3保存的地址值,是数据在堆空间中开辟空间以后对应的地址值。
面试题:
String s = new String(“abc”);方式创建对象,在内存中创建了几个对象?
两个:
- 一个是堆空间中new结构 ==> 对象名 s 类型为string
- 另一个是char[]对应的常量池中的数据"abc" ==> 对象名value 类型为char[]
解析:
- 一个是 new出来 string类型 ,对象名为s的对象,这个对象的地址值指向堆空间.
- 另一个是string对象里面属性 :private final char value[ ];char数组类型的,属性名叫value.这也是个对象,且这个对象地址值指向方法区对应的常量池中的数据
练习题 1:
@Test
public void test2(){
//通过字面量定义的方式:此时的s1和s2的数据javaEE声明在方法区中的字符串常量池中。
String s1 = "javaEE";
String s2 = "javaEE";
//通过new + 构造器的方式:此时的s3和s4保存的地址值,是数据在堆空间中开辟空间以后对应的地址值。
String s3 = new String("javaEE");
String s4 = new String("javaEE");
System.out.println(s1 == s2);//true
System.out.println(s1 == s3);//false
System.out.println(s1 == s4);//false
System.out.println(s3 == s4);//false
}
练习题 2:
@Test
public void test2(){
Person p1 = new Person("Tom",12);
Person p2 = new Person("Tom",12);
System.out.println(p1.name.equals(p2.name));//true
System.out.println(p1.name == p2.name);//true
p1.name = "Jerry";
System.out.println(p2.name);//Tom
}
- p1 对象 与 p2是两个new出来的对象 地址值指向堆中不同的区域.
- 但他们string类型的属性,属性名为Tom时.都是指向常量池中的同一区域.
- 当属性名为Jerry时,此时这个string类型的属性对象 ,指向常量池中的另一区域.
判断题:
@Test
public void test3(){
String s1 = "javaEE";
String s2 = "hadoop";
String s3 = "javaEEhadoop";
String s4 = "javaEE" + "hadoop";
String s5 = s1 + "hadoop";
String s6 = "javaEE" + s2;
String s7 = s1 + s2;
System.out.println(s3 == s4);//true
System.out.println(s3 == s5);//false
System.out.println(s3 == s6);//false
System.out.println(s3 == s7);//false
System.out.println(s5 == s6);//false
System.out.println(s5 == s7);//false
System.out.println(s6 == s7);//false
String s8 = s6.intern();//返回值得到的s8使用的常量值中已经存在的“javaEEhadoop”
System.out.println(s3 == s8);//true
}
结论:
- 常量与常量的拼接结果在常量池。且常量池中不会存在相同内容的常量。
- 只要其中有一个是变量,结果就在堆中。
- 如果拼接的结果调用intern()方法,返回值就在常量池中
练习题 3;
练习题 4;
- 这边str是两个变量,一个是成员变量,一个是局部变量
值传递:
- 基本数据类型
传的是你存储的数据 - 引用数据类型
传的是地址值