`
sarin
  • 浏览: 1758574 次
  • 性别: Icon_minigender_1
  • 来自: 大连
博客专栏
E3b14d1f-4cc5-37dd-b820-b6af951740bc
Spring数据库访问系列...
浏览量:173849
C2083dc5-6474-39e2-993e-263652d27795
Android学习笔记
浏览量:368343
5f40a095-b33c-3e8e-8891-606fcf3b8d27
iBatis开发详解
浏览量:189452
B272a31d-e7bd-3eff-8cc4-c0624ee75fee
Objective-C学习...
浏览量:100013
社区版块
存档分类
最新评论

Java String对象的经典问题(new String())

阅读更多
    先来看一个例子,代码如下:
public class Test {
	public static void main(String[] args) {
		String str = "abc";
		String str1 = "abc";
		String str2 = new String("abc");
		System.out.println(str == str1);
		System.out.println(str1 == "abc");
		System.out.println(str2 == "abc");
		System.out.println(str1 == str2);
		System.out.println(str1.equals(str2));
		System.out.println(str1 == str2.intern());
		System.out.println(str2 == str2.intern());
		System.out.println(str1.hashCode() == str2.hashCode());
	}
}

    如果您能对这8个输出结果直接判断出来,下面的分析就不用看了。但是我想还是有很多人对这个String对象这个问题只是表面的理解,下面就来分析一下Java语言String类和对象及其运行机制的问题。
    做个基础的说明,堆(heap)内存和栈(Stack)内存的问题。堆和栈的数据结构这里就不解释了。Java语言使用内存的时候,栈内存主要保存以下内容:基本数据类型和对象的引用,而堆内存存储对象,栈内存的速度要快于堆内存。总结成一句话就是:引用在栈而对象在堆。
    Java中的比较有两种,是==和equals()方法,equals()是Object类的方法,定义在Object类中的equals()方法是如下实现的:
	public boolean equals(Object obj){
		return (this==obj);
}

    String类重写了equals()方法,改变了这些类型对象相等的原则,即判断对象是否相等依据的原则为判断二者的内容是否相等。
    了解以上内容后我们来说说String,String类的本质是字符数组char[],其次String类是final的,是不可被继承的,这点可能被大多数人忽略,再次String是特殊的封装类型,使用String时可以直接赋值,也可以用new来创建对象,但是这二者的实现机制是不同的。还有一个String池的概念,Java运行时维护一个String池,池中的String对象不可重复,没有创建,有则作罢。String池不属于堆和栈,而是属于常量池。下面分析上方代码的真正含义
	String str = "abc";
	String str1= "abc";

    第一句的真正含义是在String池中创建一个对象”abc”,然后引用时str指向池中的对象”abc”。第二句执行时,因为”abc”已经存在于String池了,所以不再创建,则str==str1返回true就明白了。str1==”abc”肯定正确了,在String池中只有一个”abc”,而str和str1都指向池中的”abc”,就是这个道理。
String str2 = new String("abc");

    这个是Java SE的热点问题,众所周知,单独这句话创建了2个String对象,而基于上面两句,只在栈内存创建str2引用,在堆内存上创建一个String对象,内容是”abc”,而str2指向堆内存对象的首地址。
    下面就是str2==”abc”的问题了,显然不对,”abc”是位于String池中的对象,而str2指向的是堆内存的String对象,==判断的是地址,肯定不等了。
    str1.equals(str2),这个是对的,前面说过,String类的equals重写了Object类的equals()方法,实际就是判断内容是否相同了。
    下面说下intern()方法,在JavaDoc文档中,这样描述了intern()方法:返回字符串对象的规范化表示形式。怎么理解这句话?实际上过程是这样进行的:该方法现在String池中查找是否存在一个对象,存在了就返回String池中对象的引用。
    那么本例中String池存在”abc”,则调用intern()方法时返回的是池中”abc”对象引用,那么和str/str1都是等同的,和str2就不同了,因为str2指向的是堆内存。
    hashCode()方法是返回字符串内容的哈希码,既然内容相同,哈希码必然相同,那他们就相等了,这个容易理解。
再看下面的例子:
public class Test {
	private static String str = "abc";
	public static void main(String[] args) {
		String str1 = "a";
		String str2 = "bc";
		String combo = str1 + str2;
		System.out.println(str == combo);
		System.out.println(str == combo.intern());
	}
}

    这个例子用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,显然str==combo是错误的,而str==combo.intern()是正确的,在String池中也存在”abc”,那就直接返回了,而str也是指向String池中的”abc”对象的。此例说明任何重新修改String都是重新分配内存空间,这就使得String对象之间互不干扰。也就是String中的内容一旦生成不可改变,直至生成新的对象。
    同时问题也来了,使用+连接字符串每次都生成新的对象,而且是在堆内存上进行,而堆内存速度比较慢(相对而言),那么再大量连接字符串时直接+是不可取的,当然需要一种效率高的方法。Java提供的StringBuffer和StringBuilder就是解决这个问题的。区别是前者是线程安全的而后者是非线程安全的,StringBuilder在JDK1.5之后才有。不保证安全的StringBuilder有比StringBuffer更高的效率。
    自JDK1.5之后,Java虚拟机执行字符串的+操作时,内部实现也是StringBuilder,之前采用StringBuffer实现。
    欢迎交流,希望对使用者有用。
32
4
分享到:
评论
11 楼 赵庆辉 2015-07-04  
看帖回复是美德,楼主讲的很清晰明了,看了豁然开朗.
10 楼 abc08010051 2015-03-05  
taozhi8833998 写道
   这个例子用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,显然str==combo是错误的??
  
   请附加说明,只是在楼主的例子中+连接字符串时候,是在堆创建的对象,如果下述情况就不对了
         String c="a"+"bc";
         System.out.println(c.intern()==c);//true

楼主说在堆里面创建对象,是因为你+的前后都是一个字符串对象,而不是双引号的字符串。


层主说的,如果两个双引号的字符串相加:String c="a"+"bc"; 虚拟机在编译的时候已经把两个字符串合并成一个字符串"abc",虚拟机的常量池中没有"a"和"bc",而是一个"abc";当时两个对象用+时, String str1 = "a"; 
        String str2 = "bc"; 
        String combo = str1 + str2;
编译后的class文件的命令行为创建一个StringBuilder对象,把str1,str2用append方法连接起来以后然后调用toString方法创建一个新的String对象
9 楼 taozhi8833998 2014-10-28  
   这个例子用来说明用+连接字符串时,实际上是在堆内容创建对象,那么combo指向的是堆内存存储”abc”字符串的空间首地址,显然str==combo是错误的??
  
   请附加说明,只是在楼主的例子中+连接字符串时候,是在堆创建的对象,如果下述情况就不对了
         String c="a"+"bc";
         System.out.println(c.intern()==c);//true

楼主说在堆里面创建对象,是因为你+的前后都是一个字符串对象,而不是双引号的字符串。
8 楼 gwgyk 2014-04-18  
请问代码1中第12行“System.out.println(str2 == str2.intern());  ”中的intern()方法,是否将str2.intern()的返回值赋给str2?
7 楼 douglozy 2014-02-19  
That's good。
6 楼 liushuiwuqing4 2013-09-04  
大哥,你这解释到位得很,佩服佩服。一下子就明白了很多,谢谢。
5 楼 黑眼睛用来翻白眼 2013-08-04  
不错,很清晰
4 楼 daofeng1983 2013-07-23  
学习了,谢谢
3 楼 sarin 2010-03-03  
risemanjavaeye 写道
  自JDK1.5之后,Java虚拟机执行字符串的+操作时,内部实现也是StringBuilder,之前采用StringBuffer实现。

之前也是基于stringbuffer实现的,能说说为什么会和直接用stringbuffer有性能上的区别吗?

我们都知道StringBuffer是可变内容的,而String不是,这是很清楚的,虚拟机执行+相连字符串时,内部实现使用的是StringBuffer,但是每次相加后都在堆区重新创建了一次对象。而StringBuffer不是。
2 楼 risemanjavaeye 2010-03-03  
  自JDK1.5之后,Java虚拟机执行字符串的+操作时,内部实现也是StringBuilder,之前采用StringBuffer实现。

之前也是基于stringbuffer实现的,能说说为什么会和直接用stringbuffer有性能上的区别吗?
1 楼 peng_joy 2010-03-03  
支持,不忘java基础及底层知识的学习

相关推荐

    Java String对象的经典问题

    ### Java String对象的经典问题 #### 一、String 类与对象机制概述 在Java中,`String`类是一个非常重要的类,它提供了丰富的功能用于处理文本数据。`String`类是不可变的(immutable),这意味着一旦一个`String`...

    String对象创建问题

    在Java编程语言中,`String`对象的创建是开发者经常遇到的问题,因为它涉及到内存管理和效率。`String`类在Java中被广泛使用,因为它代表不可变的字符序列,这使得它在很多场景下非常安全。这里我们将深入探讨`...

    java String 使用详解

    本文将详细介绍 Java String 的使用方法,包括 String 对象的初始化、字符串的常见操作等。 一、String 对象的初始化 String 对象的初始化是一种特殊的语法,Java 提供了简化的初始化方法,例如: String s = ...

    Java中颜色的String和Color对象之间的互相转换

    Java 中颜色的 String 和 Color 对象之间的互相转换 在 Java 中,颜色的表示形式有多种,包括字符串形式和 Color 对象形式。这两种形式可以互相转换,本文将介绍 Java 中颜色的 String 和 Color 对象之间的互相转换...

    java中Object对象String对象的解析.pdf

    "Java 中 Object 对象和 String 对象的解析" Java 中的 Object 对象和 String 对象是两个非常重要的概念。在 Java 中,每个对象都继承自 Object 对象,这意味着每个对象都拥有 Object 对象的方法和属性。String ...

    解析Java中的String对象的数据类型 字符串

    在Java中,如果我们使用字面值方式创建的String对象,它们是常量池中的字符串常量,如果我们使用new关键字创建的String对象,它们是运行时创建的新对象。例如,String s0="kvill";,String s1=new String("kvill");...

    Java String与Byte类型转换

    在Java编程中,String对象和Byte类型的转换是常见的操作,特别是在网络编程中,因为网络通信通常涉及字节流的处理。下面将详细讲解Java中如何进行这两种类型之间的转换,并探讨其在网络编程中的应用。 首先,让我们...

    小心String的陷阱——深入剖析Java中String的处理机制

    标题和描述均强调了在Java中处理`String`对象时可能遇到的陷阱,尤其是在理解和使用`String`的处理机制上。文章由天津工业大学软件工程专业的翁龙辉撰写,旨在深入剖析`String`在Java中的独特行为及其潜在的陷阱。...

    我们一起学Java之String

    但如果使用new关键字创建了新的String对象,即使内容相同,它们也是不同的对象,引用不同的内存地址,因此使用"=="比较时会返回false,这时应该使用String类的equals()方法来判断字符串的内容是否相等。 字符串拼接...

    java String类的实现

    Java虚拟机(JVM)维护了一个字符串常量池,用于存储所有的字面量字符串和通过`new String()`创建的字符串。如果字符串已经存在于池中,`String`构造函数将不会创建新对象,而是返回池中已有的引用。 ### 7. `...

    Java中String判断值为null或空及地址是否相等的问题

    本文主要讨论了如何正确判断Java中的String对象是否为null、空值("")以及它们的地址是否相等。在处理字符串时,了解这些概念对于避免程序出错至关重要。 首先,我们需要区分`null`和空字符串`""`。`null`表示变量...

    JAVA面试题解惑系列(二)——到底创建了几个String对象-JAVA程序员JAVA工程师面试必看.pdf,这是一份不错的文件

    本文探讨了 JAVA 中 String 对象的创建机制,包括使用 new 关键字、使用引号包含的文本、使用连接符“+”等方式,并探索了 String 对象池的概念和机制。了解这些 cơ bản知识点对 JAVA 开发人员非常重要,可以帮助...

    Java中的String对象是不可变的吗Java开发Jav

    在Java编程语言中,String对象被认为是不可变的。这个特性是Java设计者为了优化性能、安全性和线程安全性而有意设定的。理解String对象的不可变性对于Java开发者来说至关重要,因为它影响着代码的编写、内存管理和多...

    JAVA_String

    6. `replace(oldChar, newChar)`或`replace(oldString, newString)`:替换字符串中符合条件的字符或子串。 比较`String`对象: - `equals(anotherString)`:比较两个字符串的内容是否相等,不考虑大小写。 - `...

    StringtoList和StringtoMap和StringtoObject和StringtoArray

    Map<String, String> map = gson.fromJson(jsonString, new TypeToken<Map<String, String>>(){}.getType()); ``` 4. **String to Object** 如果JSON字符串代表的是一个自定义Java对象,你可以创建一个对应的类...

    string 对象 与json互转

    与此相关的,`String`对象是Java编程语言中的基础类型,用于存储和处理文本数据。在实际开发中,我们经常需要在`String`对象与JSON对象之间进行转换,以满足不同的需求。本文将深入探讨`String`与JSON的互转方法,并...

    String类创建对象问题

    ### String类创建对象问题 #### 一、String 类与常量池、字符串池的基本概念 在 Java 中,`String` 类并不属于八种基本数据类型之一,而是作为一个对象存在。这意味着 `String` 对象默认值为 `null`。尽管如此,`...

    String创建对象

    // 转换为String对象 ``` 4. **intern() 方法**: `String`类的`intern()`方法用于获取常量池中的字符串引用。如果常量池中已存在该字符串,则返回其引用;否则,将字符串添加到常量池并返回引用。例如: ```...

    简单说说JAVA的String和byte[]的关系

    在Java编程语言中,`String`对象与`byte[]`数组之间的转换是常见的操作之一。理解这两者之间的关系对于处理文本数据、网络通信及文件读写等任务至关重要。 #### 一、String与byte[]的基本概念 - **String**: 在...

Global site tag (gtag.js) - Google Analytics