认识String类
创建字符串
常见的构造String方式
// 方式一
String str = "Hello Word";
// 方式二
String str2 = new String("Hello Word");
// 方式三
char[] array = {'a', 'b', 'c'};
String str3 = new String(array);
注意事项:
“Hello”这样的字面值常量,也是String类型
String也是引用类型,String str=“Hello”;堆栈分布如下
由于String是引用类型
String str="Hello";
String str2=str1;
内存布局如图
此时修改str
str="Word";
System.out.println(str2);
//结果:
//Hello
此时我们发现,修改str的值,并不会对str2有所影响 ,而是str这个应用指向了一个新的值
字符串比较相等
//如果是int变量
int a=10;
int b=10;
System.out.println(a==b);
//结果:true
String str="Hello";
String str1="Hello";
System.out.println(str==str1);
//结果:true 好像没问题,在试一个
String str=new String("Hello");
String str=new String("Hello");
System.out.println(str==str1);
//结果:false
这就涉及到另一个知识点,字符串常量池:
如 “Hello” 这样的字符串字面值常量, 也是需要一定的内存空间来存储的. 这样的常量具有一个特点, 就是不需要修改(常量嘛). 所以如果代码中有多个地方引用都需要使用 “Hello” 的话, 就直接引用到常量池的这个位置就行了, 而没必要把 “Hello” 在内存中存储两次.
但是像String str=new String(“Hello”)这样的方式创建的String对象,在堆上开辟了一个新的空间存储“Hello”,也就是有两份。
所以得出一个结论就是String中的“==”不是比较字符串的内容,而是比较两个引用是否指向同一个对象,那么字符串的比较该用什么呢?String类中提供了一个equals方法专门去比较String中的内容
String str=new String("Hello");
String str=new String("Hello");
System.out.println(str.equals(str1));
//结果:true
注意事项:
如果要让对象和字符串内容去比较的话用方式2最好,因为方式1中str如果为null的话,就会报java.lang.NullPointerException异常
String str = new String("Hello");
// 方式一
System.out.println(str.equals("Hello"));
// 方式二
System.out.println("Hello".equals(str));
常量池
String 类的设计使用了共享设计模式
在JVM底层实际上会自动维护一个对象池(字符串常量池)
如果现在采用了直接赋值的模式进行String类的对象实例化操作,那么该实例化对象(字符串内容)将自动保存到这个对象池之中.
如果下次继续使用直接赋值的模式声明String类对象,此时对象池之中如若有指定内容,将直接进行引用, 如若没有,则开辟新的字符串对象而后将其保存在对象池之中以供下次使用
String str1 = "hello" ;
String str2 = "hello" ;
String str3 = "hello" ;
System.out.println(str1 == str2); // true
System.out.println(str1 == str3); // true
System.out.println(str2 == str3); // true
本列中直接赋值的对象在常量池中,及多个引用指向同一个对象,所以为true,千万记住,直接赋值或调用String的intern()方法的才保存到常量池中。
// 该字符串常量并没有保存在对象池之中
String str1 = new String("hello") ;
String str2 = "hello" ;
System.out.println(str1 == str2);
// 执行结果: false
String str1 = new String("hello").intern() ;
String str2 = "hello" ;
System.out.println(str1 == str2);
// 执行结果:true
第一个,str1是在堆上开辟的一个空间,str2是在常量池中开辟的空间,两个引用未指向同一个对象所以为false。第二个,调用intern()方法将对象写进常量池中,两个引用指向同一个对象,为true
面试题:请解释String类中两种对象实例化的区别
- 直接赋值:只会开辟一块堆内存空间,并且该字符串对象可以自动保存在对象池中以供下次使用。
- 构造方法:会开辟两块堆内存空间,不会自动保存在对象池中,可以使用intern()方法手工入池。
String类提供了许多封装好的方法,由于数量太多,大家可以去api查看,这里就不一一介绍。