`
itkui
  • 浏览: 27064 次
  • 性别: Icon_minigender_1
  • 来自: 北京
最近访客 更多访客>>
社区版块
存档分类
最新评论

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

    博客分类:
  • J2SE
阅读更多
        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是可改变的
分享到:
评论
15 楼 laowang_27hotmail.com 2007-09-13  
细节,真的还要好好学啊
14 楼 itkui 2007-09-12  
谢了大家,String还是满耐研究的哦
13 楼 feng_gladys 2007-09-12  
<br/>
<div class='quote_div'><br/>
最后我再破除一个错误的理解: <br/>
<br/>
有人说,“使用String.intern()方法则可以将一个String类的保存到一个全局String表中,如果具有相同值的Unicode字符串已经在这个表中,那么该方法返回表中已有字符串的地址,如果在表中没有相同值的字符串,则将自己的地址注册到表中“如果我把他说的这个全局的String表理解为常量池的话,他的最后一句话,“如果在表中没有相同值的字符串,则将自己的地址注册到表中”是错的: <br/>
<br/>
看例4: <br/>
<br/>
<table border='1' align='center' width='90%'>
    <tbody>
        <tr>
            <td>String s1=new String("kvill"); <br/>
            String s2=s1.intern(); <br/>
            System.out.println( s1==s1.intern() ); <br/>
            System.out.println( s1+" "+s2 ); <br/>
            System.out.println( s2==s1.intern() ); </td>
        </tr>
    </tbody>
</table>
<br/>
结果: <br/>
<br/>
<table border='1' align='center' width='90%'>
    <tbody>
        <tr>
            <td>false <br/>
            kvill kvill <br/>
            true </td>
        </tr>
    </tbody>
</table>
<br/>
在这个类中我们没有声名一个”kvill”常量,所以常量池中一开始是没有”kvill”的,当我们调用s1.intern()后就在常量池中新添加了一个”kvill”常量,原来的不在常量池中的”kvill”仍然存在,也就不是“将自己的地址注册到常量池中”了。 <br/>
<br/>
s1==s1.intern()为false说明原来的“kvill”仍然存在; <br/>
<br/>
s2现在为常量池中“kvill”的地址,所以有s2==s1.intern()为true。 <br/>
<br/>
</div>
s1 = new String("kill");这段代码是产生两个对象,一个是“kill”,该对象存入常量池中,另一个是复制“kill”的值新产生一个对象(new)并返回给s1,所以s1=new String(“kill”);实际上已经在常量池里注册了一个“kill”。<br/>
<br/>
String构造函数:<br/>
<h3> String</h3>
<pre>public String(<a title='class in java.lang'>String</a> original)</pre>
<dl><dd>Initializes a newly created <code>String</code> object so that it  represents the same sequence of characters as the argument; in other  words, the newly created string is a copy of the argument string. Unless   an explicit copy of <code>original</code> is needed, use of this   constructor is unnecessary since Strings are immutable.
<p> </p>
<dl><dt>Parameters:</dt><dd><code>original</code> - a <code>String</code>.</dd></dl> </dd></dl>the newly created string is a copy of the argument string  这句话说明原因
12 楼 Calmfeeling 2007-09-11  
呵呵,不是太全面...
面试么,离不开String ,StringBuffer,==&&equals的区别
11 楼 xuehongliang 2007-09-11  
还是对理解string类型有帮助的。
10 楼 hwangita 2007-09-11  
写的还是可以的
9 楼 抛出异常的爱 2007-09-11  
引用
字符串字面量通过+运算符,sun的编译器会进行优化,"a"+"b"+"c"在编译后优化为"abc",而非字面量的字符串使用+运算符将产生新的String,这个真的已经讨论烂了。

这么设计有一个好处
就是当把字符串当参数传给一个方法时
不用担心这个方法会改变这个字符串的内容
public void zz(String p){
  p += "a";//这时的p与外面的那个P的地址变的不同了。
}

外面的p的内容就被保护了。
8 楼 itkui 2007-09-09  
birdjavaeye 写道
明确说明一下,第6点理论上那样,但在现在的Sun JDK的javac实现上不是那样

"a" + new String("b") + "c"

只会产生"a"、"b"、"c"和最后的"abc",没有中间的"ab"
因为sun的javac把上面代码编译成了:
new StringBuilder("a").append(new String("b")).append("c").toString()


真正遵循规范的+要这么写:
"a".concat("b").concat("c")这样才会产生"ab"这个字符串

所以实际应用中,完全不要担心"a" + "b" + "c"这样的写法
但要注意类似下面的东西:
for (String x = ...)
  s = s + x;
这个每次循环都会生成一个字符串,这时改成s.append(x)才好

知道了,谢谢!
7 楼 itkui 2007-09-09  
birdjavaeye 写道
明确说明一下,第6点理论上那样,但在现在的Sun JDK的javac实现上不是那样

"a" + new String("b") + "c"

只会产生"a"、"b"、"c"和最后的"abc",没有中间的"ab"
因为sun的javac把上面代码编译成了:
new StringBuilder("a").append(new String("b")).append("c").toString()


真正遵循规范的+要这么写:
"a".concat("b").concat("c")这样才会产生"ab"这个字符串

所以实际应用中,完全不要担心"a" + "b" + "c"这样的写法
但要注意类似下面的东西:
for (String x = ...)
  s = s + x;
这个每次循环都会生成一个字符串,这时改成s.append(x)才好

知道了,谢谢!
6 楼 dennis_zane 2007-09-09  
字符串字面量通过+运算符,sun的编译器会进行优化,"a"+"b"+"c"在编译后优化为"abc",而非字面量的字符串使用+运算符将产生新的String,这个真的已经讨论烂了。
5 楼 birdjavaeye 2007-09-09  
明确说明一下,第6点理论上那样,但在现在的Sun JDK的javac实现上不是那样

"a" + new String("b") + "c"

只会产生"a"、"b"、"c"和最后的"abc",没有中间的"ab"
因为sun的javac把上面代码编译成了:
new StringBuilder("a").append(new String("b")).append("c").toString()


真正遵循规范的+要这么写:
"a".concat("b").concat("c")这样才会产生"ab"这个字符串

所以实际应用中,完全不要担心"a" + "b" + "c"这样的写法
但要注意类似下面的东西:
for (String x = ...)
  s = s + x;
这个每次循环都会生成一个字符串,这时改成s.append(x)才好
4 楼 itkui 2007-09-09  
Eastsun 写道
想要全面&正确理解String,看<The Java™ Language Specification>是最好的方法.

谢了哥们,下了,有用的到的时候再看看。
3 楼 Eastsun 2007-09-09  
想要全面&正确理解String,看<The Java™ Language Specification>是最好的方法.
2 楼 itkui 2007-09-09  
wang20051 写道
好文,让我对String 数据类型理解更深刻
这几天面试,都是问些基础的东西,所以就在网上找找看看。
1 楼 wang20051 2007-09-09  
好文,让我对String 数据类型理解更深刻

相关推荐

    全面理解java中的String.doc

    总之,理解Java中的String类,特别是它的创建方式、常量池以及`intern()`方法的工作原理,对于编写高效、内存友好的代码至关重要。在实际编程中,根据需求合理选择创建和操作字符串的方法,可以避免不必要的内存开销...

    Java的String类

    在Java编程中,`String`类是最常用的数据类型之一,用于处理文本数据。它是一个不可变类,意味着一旦创建了一个`String`对象,其内容就不能更改。这使得`String`类在多线程环境中非常安全,同时也带来了一些性能上的...

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

    Java中的`String`对象是一种非常特殊的数据类型,虽然它不属于八种基本数据类型(byte, short, int, long, char, float, double, boolean),但它却在编程中扮演着重要角色。`String`是一个对象,这意味着它在内存中...

    oracle-java数据类型1

    在 Oracle 数据库与 Java 应用程序交互时,理解 SQL 数据类型和它们对应的 JDBC 类型以及 Oracle 扩展的 Java 类型是至关重要的。这里我们将深入探讨这些概念,并提供一个全面的指南。 首先,JDBC(Java Database ...

    Java中关于String的全面解析

    String Pool串池是在内存堆中专门划分一块空间,用来保存所有String对象数据,当构造一个新字符串String对象时(通过字面量赋值的方法),Java编译机制会优先在这个池子里查找是否已经存在能满足需要的String对象,...

    2016年java基本数据类型介绍.docx

    ### 2016年Java基本数据类型介绍 #### 一、引言 Java作为一门广泛应用的编程语言,...通过本文的介绍,相信读者已经对Java的内置数据类型有了较为全面的认识,并能够根据实际需求选择合适的数据类型来完成程序设计。

    JAVA基础之基本数据类型全面解析

    本文将对Java中的基本数据类型进行全面解析,包括数值型、字符型、布尔型以及字符串。 一、数值型 数值型数据用于表示各种数值,包括整数和浮点数。 1)整数: - byte:占用1个字节,8位,可存储的数值范围是0到...

    characterString_java_in_character_string_

    在Java编程语言中,字符(Character)是基本的数据类型之一,用于处理单个字符。`characterString_java_in_character_string_`这个标题暗示我们将探讨如何在Java中处理字符字符串,特别是查找字符串中的特定字符。...

    java中文解释的API

    首先,Java API中的核心类库包括了基础数据类型、对象模型、异常处理、字符串操作、集合接口和实现(如ArrayList、HashMap)、I/O流、网络编程、日期时间处理等。这些类库为开发者提供了构建复杂应用的基础模块。 ...

    字符转换工具类

    在HTML生成过程中,有时需要将数字数据转化为特定格式的字符串,以便在网页上显示,或者在处理用户输入时,将表单中的字符串值转换回数值类型进行计算。 一个全面的字符转换工具类通常会包括以下功能: 1. **编码...

    java String源码和String常量池的全面解析

    Java String 源码和 String 常量池的全面解析 Java String 源码和 String 常量池是 Java 语言中非常重要的两个概念,它们之间存在着紧密的联系。在 Java 语言中,String 类是不可变的,finalize 方法被禁用,以确保...

    SuperMap iObjects Java 8C后台数据的增删改查

    运行这个文件,可以帮助理解如何在实际项目中使用SuperMap iObjects Java 8C进行数据操作。阅读`范例代码文档.pdf`将提供更详细的步骤和代码示例。 综上所述,SuperMap iObjects Java 8C为开发者提供了全面的工具来...

    重新理解Java泛型

    通过深入理解Java泛型,开发者可以编写更安全、更具可读性和可维护性的代码,同时避免不必要的类型转换和潜在的运行时错误。对于Java集合框架的使用,泛型是不可或缺的一部分,它使得集合能够以类型安全的方式存储和...

    Java语言程序设计与数据结构(第11版)_java_

    1. **Java编程基础**:本书首先会介绍Java语言的基础语法,包括变量、常量、数据类型(如整型、浮点型、字符型、布尔型以及引用类型)、运算符、流程控制(条件语句、循环语句)以及异常处理。 2. **面向对象编程**...

    Java基础,Java进阶,Java数据结构,十大算法

    1. **基本语法**:Java的语法规则,如变量声明、数据类型(整型、浮点型、字符型、布尔型等)、运算符、流程控制(条件语句、循环语句)以及异常处理。 2. **类与对象**:面向对象编程的基础,包括类的定义、对象的...

    java面试.doc很全面的哦

    - 引用数据类型:如String、Integer,是对象的引用,指向内存中的对象地址。String是不可变的,而Integer是int的包装类,提供了额外功能。 3. String与StringBuilder/StringBuffer: - String是不可变的,每次...

    Java类库 java中常用的类 可以参考

    Java 类库为开发者提供了丰富的工具集,涵盖了从基础的数据类型处理到高级的网络通信等多个方面。本文档旨在介绍 Java 中一些常用的类库,帮助初学者更好地理解和使用这些类库。尽管文档内容并不全面,但包含了...

    全面解析Java支持的数据类型及Java的常量和变量类型

    Java编程语言提供了两种主要的数据类型:原始数据类型和引用数据类型。...总的来说,理解Java的数据类型和常量对于编写有效的Java代码至关重要,它们决定了程序如何存储和处理数据,以及如何进行各种运算和逻辑判断。

    Java理解程序逻辑(基础篇)

    2. **基本语法**:Java的基础包括变量、数据类型、运算符、控制结构(如if语句、for循环、while循环)以及函数。学习者需要了解如何声明、初始化和使用这些元素来控制程序流程。 3. **面向对象编程**:Java是面向...

Global site tag (gtag.js) - Google Analytics