1.
public class Str {
public static void main( String[]args ) {
//TODOAuto-generated method stub
StringStringa="hello";
StringStringb=newString("hello");
StringStringc=Stringb;
System.out.println(Stringa==Stringb);//false
System.out.println(Stringa==Stringc);//false
System.out.println(Stringb==Stringc);//true
System.out.println(Stringa.equals(Stringb));//true
System.out.println(Stringa.equals(Stringc));//true
System.out.println(Stringb.equals(Stringc));//true
}
}
“==”比较的并不是字符串包含的内容,而是将字符存放的地址进行比较,属于数值比较
比较字符串里面的内容,可以使用String里面定义的方法
---比较内容(与原定义有一些差别):public boolean equals (String str)
2. Stringinput=null;//假设为用户输入 如果(input.equals("hello"),就会产生空指向异常
if("hello".equals(input)) {//equals处理了null
System.out.println("Hello world!!");
//小技巧:在开发中,如果要要判断输入的内容是否是某一字符串,一定要将字符串放在前面
}
3.
Stringstra="hello";
Stringstrb="hello";
Stringstrc="hello";
Stringstrd="world";
System.out.println(stra==strb);//true
System.out.println(stra==strc);//true
System.out.println(strb==strc);//true
System.out.println(stra==strd);//false
如图所示
共享设计模式
(1)在JVM的层实际上会存在有一个对象池(不一定只保存 string对象) ,当代码之中采用直接赋值的方式定义了一个string类对象时,会将此字符串对象所使用的匿名对象入池保存, 而后如果后续还有其他string类对象也采用了相同赋值方式,且所设置同样内容时,那么将不会开辟新的堆内存空间,而是使用已有的对象进行引用的分配,从而继续使用
(2)采用构造方法实例化
构造方法如果要使用关键字new,一旦使用了关键字new,就表示要开辟新的堆内存
Stringstr=newString("hello");
Stringstra="hello";
System.out.println(stra==str);//false
通过上图可知,如果使用的时构造方法string对象实例化的时候,那么最终的操作形式就变为了开辟两块堆内存空间(并且有一块将成为了垃圾空间)。
除了内存的浪费之外,如果使用了构造方法定义的string类对象,其内容不会保存在对象池之中,因为是使用了关键字new开辟的新内存,如果希望现在开辟的堆内存也可以进行对对象池的保存,可以进行手工入池:
public String intern();
Stringstr=newString("hello").intern();
Stringstra="hello";
System.out.println(stra==str);//true
面试题:请解释String对象两种实例化的区别
-- 1 直接赋值(Stringstra="字符串"),只会开辟一块堆内存空间,并会保存在对象池中以供下次重用
-- 2 构造方法 (Stringstr=newString("字符串")),会开辟两块堆内存空间,并且有一块将成为垃圾,并且不会自动入池,但是可以手动使用intern方法手工入池