String类两种实例化方式区别
1、分析直接赋值的对象实例化模式
在程序之中只需要讲一个字符串赋值给String类的对象就可以实现对象的实例化处理。
public class name {
public static void main(String args[]) {
String str = "mldn";
}
}
这种情况下肯定只会开辟出一块堆内存空间 内存空间如下
除了这种内存模式之外,利用直接赋值实例化String的形式还可以实现同一个字符串对象数据的共享操作。
观察String直接赋值时的数据共享
public class name {
public static void main(String args[]) {
String strA = "mldn";
String strB = "mldn";
System.out.println(strA == strB);//地址判断 //true
}
}
此时程序的判断结果返回了“true”,那么可以得出结论,这两个对象所指向的堆内存是同一个,也就是说这个时候内存关系图如下:
之所以现在会有这样的特点,主要的原因是因为在Java程序的底层里面提供有一个专门的字符串池(字符串数组)
观察字符串池
在采用直接赋值的处理过程之中,对于字符串而言可以实现池数据的自动保存,这样如果再有相同数据定义时可以减少对象的产生,以提升我们的操作性能。
也就是 直接赋值的操作会进行入池以进行重用处理
2、构造方法实例化
public class name {
public static void main(String args[]) {
String str = new String("mldn");
}
}
此时会开辟两块堆内存空间,而后只会使用一块,而另一块由于字符串常量所定义的匿名对象将成为垃圾空间。
观察另一种形式
public class name {
public static void main(String args[]) {
String strA = "mldn";
String strB = new String("mldn");
}
}
一个是内存中开辟新的字符串对象,另外一个是实现对象的重用。
除了以上的特点之外,在使用构造方法实例化String类对象时不会自动出现保存到字符串池的处理。
观察构造方法实例化对象时的操作
public class name4 {
public static void main(String args[]) {
String strA = "mldn";
String strB = new String("mldn");
System.out.println(strA == strB); //false
}
}
可以发现构造方法实例化的对象实际上是属于一种自己专用的内存空间,但是在String类里面也提供有帮助开发者实现手工入池的处理情况。 这个方法:public String intern();
public class name4 {
public static void main(String args[]) {
String strA = "mldn";
String strB = new String("mldn").intern();//代码链
System.out.println(strA == strB);//true
}
}
在使用构造方法定义对象之后由于使用了intern()方法,所以即便是构造出来的String类对象的内容也可以实现对象池的统一管理,但是这种做法太啰嗦。
面试题:请解释String类两种构造方法实例化方法的区别?
- 直接赋值:只会产生一个实例化对象,并且可以自动保存到对象池之中,以实现该字符串实例的重用
- 构造方法:会产生两个实例化对象,并且不会自动入池,无法实现对象重用,但是可以利用intern()方法手工入池处理。
所以以后碰见String 不要用构造方法, 用直接赋值是最稳妥的。