java

位置:IT落伍者 >> java >> 浏览文章

全面解析Java中的String数据类型


发布日期:2018年10月16日
 
全面解析Java中的String数据类型
首先String不属于种基本数据类型String是一个对象

因为对象的默认值是null所以String的默认值也是null但它又是一种特殊的对象有其它对象没有的一些特性

new String()和new String()都是申明一个新的空字符串是空串不是null

String str=kvillString 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为truekvill也都是字符串常量当一个字符串由多个字符串常量连接而成时它自己肯定也是字符串常量所以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的字符串并返回它的引用看例就清楚了

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的实例一旦生成就不会再改变了比如说String str=kv+ill+ +ans;

就是有个字符串常量首先kvill生成了kvill存在内存中然后kvill又和 生成 kvill 存在内存中最后又和生成了kvill ans;并把这个字符串的地址赋给了str就是因为String的不可变产生了很多临时变量这也就是为什么建议用StringBuffer的原因了因为StringBuffer是可改变的

上一篇:struts中文问题和国际化问题的终极解决方案

下一篇:java唯一的字符串生成器