在java中,不同的字符串赋值方法,其所在的地址可能不同也就导致,两个字符串的值看似相等可是在s1==s2操作时,其结果返回的却是false
例:
String s1 = "Programming";
String s2 = new String("Programming");
String s5 = "Program" + "ming";
System.out.println(s1 == s2); //返回值为false
System.out.println(s1 == s5);//返回值为true
其原因就与java中内存分配的机制有关:而当一个字符串被定义时,会先到常量池中寻找,如果常量池中有这个字符串,则地址直接指向常量池中字符串,如常量池中不存在则将其放入到常量池中;
关于java的内存分配机制:
https://www.cnblogs.com/dingyingsi/p/3760730.html
http://blog.csdn.net/tutngfei1129287460/article/details/7383480
例:
//常量池中本来就不含有jva字符,所以new的jva会放入常量池,属于同一个对象,返回true 先new一个SringBuffer对象,此时单独开辟一个区域,然后调用append方法
//拼接字符串(相当于对字符串的拼接而不是new的对象了),将拼接完成的字符串放入常量池,此时new出来的对象就是需要放入常量池的,返回true
String s1 = new StringBuilder("j").append("va").toString();
System.out.println(s1.intern()==s1);//true
//常量池中本身就有字符串java所以返回的并不是同一个字符串 s2.intern()返回常量池中的
//java s2为new的字符所以不相等返回false
String s2 = new StringBuilder("ja").append("va").toString();
System.out.println(s2.intern()==s2);//false
//返回false因为new出来的对象单独开辟一块区域放在java堆中,与常量池中的不同和
String s3 = new StringBuilder("Programming").toString();
System.out.println(s3.intern()==s3);//false
public static void main(String[] args){
String s1 = "Programming";
String s2 = new String("Programming");
String s3 = "Program";
String s4 = "ming";
String s5 = "Program" + "ming";
String s6 = s3 + s4;
String s7 = new StringBuilder("Programming").toString();
String s8 = new StringBuilder("Program").append("ming").toString();
System.out.println(s1 == s2); //false s1被赋予字符串Programming后发现常量池中并没有这一字符串,就将其
//放入常量池
System.out.println(s1 == s5);//true //字符串的+操作其本质是创建了StringBuilder对象进行append操作,
//然后将拼接后的StringBuilder对象用toString方法处理成String对象,s5拼接完成后发现常量池中已经有字符串Programming
//所以s5直接取常量池中的字符串所以返回true
System.out.println(s1 == s6);//false 这个暂时没搞懂
System.out.println(s1 == s5.intern());//true
System.out.println(s1 == s6.intern());//true
System.out.println(s2 == s2.intern());//false 因为s2 new了一个字符串对象,在堆中单独开辟一块区域,并不是常量池
//中对象 所以返回false
System.out.println(s7.intern()==s7);//false //返回false因为new出来的对象单独开辟一块区域放在java堆中,与常量池中的不同
System.out.println(s8.intern()==s8);///false 这个单独执行时是true 放在这里一起执行是false,个人推测
//是因为常量池中已经存在字符串Programming,而s8是new出来的,他拼接完成之后,其地址与常量池中字符串的地址
//不同,返回false
}