,String是一个对象。蒃因为对象的默认值是null,所以String的默认值也是null;但它又是一种特殊的对象,有其它对象没有的一些特性。()和newString(“”)都是申明一个新的空字符串,是空串不是null;=”kvill”与Stringstr=newString(“kvill”)的区别:蚃在这里,我们不谈堆,也不谈栈,只先简单引入常量池这个简单的概念。袂常量池(constantpool)指的是在编译期被确定,。它包括了关于类、方法、接口等中的常量,也包括字符串常量。肈看例1:肆Strings0=”kvill”;节Strings1=”kvill”;莈Strings2=”kv”+“ill”;袆s0==s1);膄s0==s2);螁结果为:肈true袇true芃首先,我们要知道Java会确保一个字符串常量只有一个拷贝。膀因为例子中的s0和s1中的”kvill”都是字符串常量,它们在编译期就被确定了,所以袈s0==s1为true;而”kv”和”ill”也都是字符串常量,当一个字符串由多个字符串常量连罿接而成时,它自己肯定也是字符串常量,所以s2也同样在编译期就被解析为一个字符串常蚅量,所以s2也是常量池中”kvill”的一个引用。薀所以我们得出s0==s1==s2;蕿用newString()创建的字符串不是常量,不能在编译期就确定,所以newString()螆创建的字符串不放入常量池中,它们有自己的地址空间。螃看例2:芃Strings0=”kvill”;荿Strings1=newString(”kvill”);袇Strings2=”kv”+newString(“ill”);膆s0==s1);蚂s0==s2);聿s1==s2);蚅结果为:芄false膂false螀false蚆例2中s0还是常量池中”kvill”的应用,s1因为无法在编译期确定,所以是运行时创莂建的新对象”kvill”的引用,s2因为有后半部分newString(“ill”)所以也无法在编译薁期确定,所以也是一个新创建对象”kvill”的应用;明白了这些也就知道为何得出此结果薀了。():螅再补充介绍一点:,在运行期被JVM装载,并且可以扩充。String的intern()方法就是扩充常量池的一个方法;当一个String实例str调用intern(羁)方法时,Java查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用,如果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用;看例3就清楚了芁例3:薅Strings0=“kvill”;袃Strings1=newString(”kvill”);莀Strings2=newString(“kvill”);肁s0==s1);薆“**********”);();肃s2=();//把常量池中“kvill”的引用赋给s2蒇s0==s1);蚈s0==());莄s0==s2);薃结果为:芈false蒅**********蒂false//(),但它的返回值没有赋给s1羂true//()返回的是常量池中”kvill”的引用羈true蒆最后我再破除一个错误的理解:袅有人说,“()方法则可以将一个String类的保存到一个全局Strin莁g表中,如果具有相同值的Unicode字符串已经在这个表中,那么该方法返回表中已有字符串的地址,如果在表中没有相同值的字符串,则将自己的地址注册到表中“如果我把他说螈的这个全局的String表理解为常量池的话,他的最后一句话,“如果在表中没有相同值的薈字符串,则将自己的地址注册到表中”是错的:羃看例4:螁Strings1=newString("kvill");葿Strings2=();莅s1==());莆s1+""+s2);芀s2==());艿结果:蒆false蒄kvillkvill羄true羀在这个类中我们没有声名一个”kvill”常量,()后就在常量池中新添加了一个”kvill”常量,原来的不在常量池中的”kvill”仍然存在,也就不是“将自己的地址注册到常量池中”了。蒈 s1==()为false说明原来的“kvill”仍然存在;薂 s2现在为常量池中“kvill”的地址,所以有s2==()为true。
java中String的“”和equals判断相等性 来自淘豆网m.daumloan.com转载请标明出处.