String:字符串,使用一对“”引起来表示
- String声明为final的,不能被继承
- String实现了Serializable接口,表示字符串是支持序列化的
- 实现了Comparable接口,表示String可以比较大小
- String在内部定义了final char[] value用于存储字符串数据
- String代表不可变的字符序列,简称:不可变性
- 我们通过字面量的方式给一个字符串赋值,此时的字符串值声明在字符串常量池中
- 字符串常量池中是不会存储相同内容的字符串的
package com.atguigu.java;
import org.junit.Test;
public class StringTest {
@Test
public void test1() {
String s1 = "abc";//字面量的定义方式
String s2 = "abc";
System.out.println(s1 == s2);//true比较的是两个字符串的地址值
s1 = "hello";
System.out.println(s1);//hello
System.out.println(s2);//abc
System.out.println("==================");
String s3 = "abc";
s3 += "def";
System.out.println(s2);//abc
System.out.println(s3);//abcdef
System.out.println("==================");
String s4 = "abc";
String s5 = s4.replace('a', 'm');
System.out.println(s4);//abc
System.out.println(s5);//mbc
}
}
字符串不可变性的体现
- 当对字符串重新赋值时,需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 当对现有的字符串进行连接操作时,也需要重新指定内存区域赋值,不能使用原有的value进行赋值
- 当调用String的replace()方法修改指定的字符或字符串时,不能使用原有的value进行赋值
String的实例化方式
- 方式一:通过字面量定义的方式
- 方式二:通过new + 构造器的方式
- 面试题:String s = new String(“abc”);方式创建对象,在内存中创建了几个对象?
答案:两个:一个是堆空间中new结构,另一个是char[]对应的常量池中的数据:“abc”
package com.atguigu.java;
import org.junit.Test;
public class StringTest {
@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
System.out.println("====================");
Person person = new Person("Tom", 12);
Person person1 = new Person("Tom", 12);//这里的Tom是字符串常量池中的Tom
System.out.println(person.getName().equals(person1.getName()));//true
System.out.println(person.getName() == person1.getName());//true
}
}
public class Person {
private String name;
private int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
字面量定义的字符串与拼接的字符串的区别
package com.atguigu.java;
import org.junit.Test;
public class StringTest {
@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(s5 == s6);//false
System.out.println(s5 == s7);//false
System.out.println(s6 == s7);//false
String s8 = s5.intern();//返回值得到的s8使用的常量值中已经存在的"javaEEhadoop"
System.out.println(s3 == s8);//true
}
}
结论:
- 常量与常量的拼接结果在常量池,且常量池中不会存在相同内容的常量
- 只要其中一个是变量,结果就在堆中
- 如果拼接的结果调用intern()方法,返回值就在常量池中