首先String不属于种基本数据类型String是一个对象 因为对象的默认值是null所以String的默认值也是null 但它又是一种特殊的对象有其它对象没有的一些特性 new String()和new String()都是申明一个新的空字符串是空串不是null String str=kvill String str=new String (kvill);的区别 在这里我们不谈堆也不谈栈只先简单引入常量池这个简单的概念 常量池(constant pool)指的是在编译期被确定并被保存在已编译的class文件中的一些数据它包括了关于类方法接口等中的常量也包括字符串常量 看例 String s=kvill; String s=kvill; String s=kv + ill; Systemoutprintln( s==s ); Systemoutprintln( s==s ); 结果为 true true 首先我们要知道Java会确保一个字符串常量只有一个拷贝 因为例子中的s和s中的kvill都是字符串常量它们在编译期就被确定了所以s==s为true而kv和ill也都是字符串常量当一个字符串由多个字符串常量连接而成时它自己肯定也是字符串常量所以s也同样在编译期就被解析为一个字符串常量所以s也是常量池中kvill的一个引用 所以我们得出s==s==s; 用new String() 创建的字符串不是常量不能在编译期就确定所以new String() 创建的字符串不放入常量池中它们有自己的地址空间 看例 String s=kvill; String s=new String(kvill); String s=kv + new String(ill); Systemoutprintln( s==s ); Systemoutprintln( s==s ); Systemoutprintln( s==s ); 结果为 false false false 例中s还是常量池中kvill的应用s因为无法在编译期确定所以是运行时创建的新对象kvill的引用s因为有后半部分new String(ill)所以也无法在编译期确定所以也是一个新创建对象kvill的应用;明白了这些也就知道为何得出此结果了 Stringintern() 再补充介绍一点存在于class文件中的常量池在运行期被JVM装载并且可以扩充String的intern()方法就是扩充常量池的一个方法当一个String实例str调用intern()方法时Java查找常量池中是否有相同Unicode的字符串常量如果有则返回其的引用如果没有则在常量池中增加一个Unicode等于str的字符串并返回它的引用看例就清楚了 [Page] 例 String s= kvill; String s=new String(kvill); String s=new String(kvill); Systemoutprintln( s==s ); Systemoutprintln( ********** ); sintern(); s=sintern(); //把常量池中kvill的引用赋给s Systemoutprintln( s==s); Systemoutprintln( s==sintern() ); Systemoutprintln( s==s ); 结果为 false ********** false //虽然执行了sintern()但它的返回值没有赋给s true //说明sintern()返回的是常量池中kvill的引用 true 最后我再破除一个错误的理解 有人说使用Stringintern()方法则可以将一个String类的保存到一个全局String表中 如果具有相同值的Unicode字符串已经在这个表中那么该方法返回表中已有字符串的地址 如果在表中没有相同值的字符串则将自己的地址注册到表中 如果我把他说的这个全局的String表理解为常量池的话他的最后一句话如果在表中没有相同值的字符串则将自己的地址注册到表中是错的 看例 String s=new String(\kvill\); String s=sintern(); Systemoutprintln( s==sintern() ); Systemoutprintln( s+\ \+s ); Systemoutprintln( s==sintern() ); 结果 false kvill kvill true 在这个类中我们没有声名一个kvill常量所以常量池中一开始是没有kvill的 当我们调用sintern()后就在常量池中新添加了一个kvill常量原来的不在常量池中的kvill仍然存在也就不是将自己的地址注册到常量池中了 s==sintern()为false说明原来的kvill仍然存在 s现在为常量池中kvill的地址所以有s==sintern()为true 关于equals()和==: 这个对于String简单来说就是比较两字符串的Unicode序列是否相当如果相等返回true;而==是比较两字符串的地址是否相同也就是是否是同一个字符串的引用 关于String是不可变的 这一说又要说很多大家只要知道String的实例一旦生成就不会再改变了 [Page] 比如说String str=kv+ill+ +ans; 就是有个字符串常量首先kv和ill生成了kvill存在内存中然后kvill又和 生成 kvill 存在内存中最后又和生成了kvill ans;并把这个字符串的地址赋给了str就是因为String的不可变产生了很多临时变量这也就是为什么建议用StringBuffer的原因了因为StringBuffer是可改变的 |