`
laiseeme
  • 浏览: 124727 次
  • 性别: Icon_minigender_2
  • 来自: 沈阳
社区版块
存档分类
最新评论

java String的迷惑

阅读更多
昨天晚上我看一些java的资料,碰到这样一个论断:

问:String s = new String("xyz");创建了几个String Object?
答:两个

Demo:

package test;

public class StringTest {
 /**
  * @param args
  * @author dougq
  */
 public static void main(String[] args)
 {
  String a = "xyz";
  String b= "xyz";
  System.out.println(a==b);
 
  String a2 = new String("xyz");
  String b2= new String("xyz");
  System.out.println(a2==b2);
 
 }
}

Result:

true
false

到网上查了好些资料,最终觉得下面这篇文章是一个最为满意的答复,全文如下:

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

    1. 首先String不属于8种基本数据类型,String是一个对象。

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

2. new String()和new String(“”)都是申明一个新的空字符串,是空串不是null;

3. String str=”kvill”;
String str=new String (“kvill”);的区别:

在这里,我们不谈堆,也不谈栈,只先简单引入常量池这个简单的概念。

常量池(constant pool)指的是在编译期被确定,并被保存在已编译的.class文件中的一些数据。它包括了关于类、方法、接口等中的常量,也包括字符串常量。

看例1:


String s0=”kvill”;
String s1=”kvill”;
String s2=”kv” + “ill”;
System.out.println( s0==s1 );
System.out.println( s0==s2 );


结果为:


true
true


首先,我们要知道Java会确保一个字符串常量只有一个拷贝。

因为例子中的s0和s1中的”kvill”都是字符串常量,它们在编译期就被确定了,所以s0==s1为true;而”kv”和”ill”也都是字符串常 量,当一个字符串由多个字符串常量连接而成时,它自己肯定也是字符串常量,所以s2也同样在编译期就被解析为一个字符串常量,所以s2也是常量池中” kvill”的一个引用。

所以我们得出s0==s1==s2;

用new String() 创建的字符串不是常量,不能在编译期就确定,所以new String() 创建的字符串不放入常量池中,它们有自己的地址空间。

看例2:


String s0=”kvill”;
String s1=new String(”kvill”);
String s2=”kv” + new String(“ill”);
System.out.println( s0==s1 );
System.out.println( s0==s2 );
System.out.println( s1==s2 );


结果为:


false
false
false


例2中s0还是常量池中”kvill”的应用,s1因为无法在编译期确定,所以是运行时创建的新对象”kvill”的引用,s2因为有后半部分new String(“ill”)所以也无法在编译期确定,所以也是一个新创建对象”kvill”的应用;明白了这些也就知道为何得出此结果了。

4. String.intern():

再补充介绍一点:存在于.class文件中的常量池,在运行期被JVM装载,并且可以扩充。String的intern()方法就是扩充常量池的一个方 法;当一个String实例str调用intern()方法时,Java查找常量池中是否有相同Unicode的字符串常量,如果有,则返回其的引用,如 果没有,则在常量池中增加一个Unicode等于str的字符串并返回它的引用;看例3就清楚了

例3:


String s0= “kvill”;
String s1=new String(”kvill”);
String s2=new String(“kvill”);
System.out.println( s0==s1 );
System.out.println( “**********” );
s1.intern();
s2=s2.intern(); //把常量池中“kvill”的引用赋给s2
System.out.println( s0==s1);
System.out.println( s0==s1.intern() );
System.out.println( s0==s2 );


结果为:


false
**********
false //虽然执行了s1.intern(),但它的返回值没有赋给s1
true //说明s1.intern()返回的是常量池中”kvill”的引用
true


最后我再破除一个错误的理解:

有人说,“使用String.intern()方法则可以将一个String类的保存到一个全局String表中,如果具有相同值的 Unicode字符串已经在这个表中,那么该方法返回表中已有字符串的地址,如果在表中没有相同值的字符串,则将自己的地址注册到表中“如果我把他说的这 个全局的String表理解为常量池的话,他的最后一句话,“如果在表中没有相同值的字符串,则将自己的地址注册到表中”是错的:

看例4:


String s1=new String("kvill");
String s2=s1.intern();
System.out.println( s1==s1.intern() );
System.out.println( s1+" "+s2 );
System.out.println( s2==s1.intern() );


结果:


false
kvill kvill
true


在这个类中我们没有声名一个”kvill”常量,所以常量池中一开始是没有”kvill”的,当我们调用s1.intern()后就在常量池中新添加了一个”kvill”常量,原来的不在常量池中的”kvill”仍然存在,也就不是“将自己的地址注册到常量池中”了。

s1==s1.intern()为false说明原来的“kvill”仍然存在;

s2现在为常量池中“kvill”的地址,所以有s2==s1.intern()为true。

5. 关于equals()和==:

这个对于String简单来说就是比较两字符串的Unicode序列是否相当,如果相等返回true;而==是比较两字符串的地址是否相同,也就是是否是同一个字符串的引用。

6. 关于String是不可变的

这一说又要说很多,大家只要知道String的实例一旦生成就不会再改变了,比如说:String str=”kv”+”ill”+” “+”ans”;
就是有4个字符串常量,首先”kv”和”ill”生成了”kvill”存在内存中,然后”kvill”又和” “ 生成 ”kvill “存在内存中,最后又和生成了”kvill ans”;并把这个字符串的地址赋给了str,就是因为String的“不可变”产生了很多临时变量,这也就是为什么建议用StringBuffer的原 因了,因为StringBuffer是可改变的。
分享到:
评论
1 楼 laiseeme 2007-07-18  
转别人的 呵呵 学习学习

相关推荐

    java-意想不到的迷惑

    Java编程语言中存在一些容易让人迷惑的特性,这些特性可能会导致程序出现意想不到的错误。本文将探讨两个具体的Java谜题,一个是关于整数取余操作的奇数性判断,另一个涉及浮点数精度问题。 首先,让我们来看第一个...

    Java内存分配和String类型的深度解析

     在java语言的所有数据类型中,String类型是比较特殊的一种类型,同时也是面试的时候经常被问到的一个知识点,本文结合java内存分配深度分析关于String的许多令人迷惑的问题。下面是本文将要涉及到的一些问题,...

    Java91个迷惑的问题

    4. **字符串操作**:Java中的`String`类有许多方法,如`concat()`, `substring()`, `trim()`等,理解它们的用法和行为差异是必要的。 5. **数组与集合**:Java中的数组和集合(如ArrayList, HashSet, HashMap等)的...

    Java解惑PPT7

    Java编程中的迷惑问题,或者称为Puzzlers,是学习Java时常常遇到的陷阱和微妙之处。这些Puzzlers有助于深入理解Java的语法规则和运行机制。以下将详细解析给出的Puzzles。 首先,我们来看Puzzle 66:一件私事。这个...

    java 参数传递

    Java 参数传递机制详解 Java 中的参数传递机制是一个经久不息的讨论...Java 参数传递机制详解,揭开了 Java 中的参数传递机制的奥秘,详细分析了基本类型和对象作为参数传递的机制,并揭开了对象作为参数传递的迷惑。

    解析Java泛型的类型擦除.pdf

    解析Java泛型的类型擦除 Java 泛型是 Java SE 1.5 的新特性,它们在语法和应用环境上与 ...本文将详细介绍 Java 泛型的类型擦除,解决泛型中常见的令人迷惑的问题,并为开发者提供了一个清晰的理解 Java 泛型的机制。

    java解惑

    本文将深入探讨在Java学习过程中常见的两个迷惑点:表达式的正确性和浮点数的精度问题。 首先,我们来看“Java 谜题 1——表达式谜题”。在这个谜题中,我们有一个`isOdd`方法,用于判断传入的整数`i`是否为奇数。...

    软件技术培训机构有哪些坑.pdf

    总的来说,选择IT培训机构时,要避免被营销策略迷惑,关注教学质量、课程内容和实际需求,同时,对于Java等编程语言的学习,要注重基础知识的掌握和实践经验的积累。在面试和找工作时,对语言特性的理解以及实际应用...

    JavaI/O深入学习之输入和输出

    因此,Java 中“流”类库让人迷惑的主要原因就在于:创建单一的结果流,却需要创建多个对象。I/O 需要应对的场景往往是多样化的,Java 类库的设计者则是通过创建大量的类来解决这个难题,区区一篇文章难以详述,本文...

    第5周 章节测试 课后实验(拓展提优)

    让你永远不再因Collections.sort()而迷惑(1) 【问题描述】设计一个学生类student,记录学生学号、姓名以及数学、英语、Java课程成绩,并可根据三门课程平均分进行排序(实现comparable接口)。创建一个学生list,...

    JavaScript 学习技巧

    String.prototype.replace 函数经常会让那些非常熟悉 C# 或者 Java 的程序员感到迷惑。可以使用正则表达式来实现字符串替换,并且可以指定在替换时忽略大小写。 五、将 arguments 转化为数组 函数中的预定义变量 ...

Global site tag (gtag.js) - Google Analytics