看了很多文章事实证明之前的通过new创建String对象只有一个是错误的,实际上创建一个或者创建两个对象,一个在堆区,一个在常量池,当常量池中已经存在就不会创建。看了一篇非常好的文章http://www.cnblogs.com/wxgblogs/p/5635099.html,详细讲述了创建String对象的过程和String中的intern方法,以及该方法在jdk1.6和1.7中intern方法的变化
/********************************************************************2017/9/21修改************************************************************************/
String类是被final修饰的类,没有子类,不能被继承,被final修饰的类我们不能断定他就是不变类,StringBuffer类也是被final修饰的类,他就是可变类,可变类与不可变类要看他内部怎么实现的,String底层声明了一个名为value的字符数组,注意这个数组使用final修饰的,String的不可变就是因为他,只要创建了字符串也就是向这个数组中存入内容了,那么这个引用指向的内容就不变。而StringBuffer的创建依赖继承的抽象类AbstractStringBuilder,他的创建默认只是创建了一个16个字符长度的数组,这个数组并没有用final修饰,所以StringBuffer是一个可变类。


        String str = "a";
String str1 = str + "b";
System.out.println(str == str1);

Java 会确保一个字符串常量只有一个拷贝。

如果String是一个可变类,也就是说可以对原对象操作,那么上面的输出应该是true,但是结果是false。也就说明了String对象一旦创建就不可以改变,那么上面的代码怎么运行的呢?简单说一下:用户通过直接赋值,在常量池中创建了一个字符串a,后面想对这个a进行追加操作,加上一个b,但是String不允许直接在原来对象后面追加,这时候就需要,再创建一个新的常量吧a拷贝过去的同时后面放上b,这个ab字符串常量是一个整体,和最开始的那个a是互不相关的。所以通过==判断地址值的时候返回结果是false。

注意:这个被final关键字修饰的只是引用不可变,也就是他指向的对象不能变,至于对象中的内容,随意,看下面
        final StringBuffer str = new StringBuffer("a");
str.append("b");//可以运行 str = new StringBuffer("a");//报错
看下面例子

        String s1 = "ab";
String s2 = "b";
String s3 = "a" + s2;
System.out.println(s1 == s3); //false String s1 = "ab";
final String s2 = "b";
String s3 = "a" + s2;
System.out.println(s1 == s3); //true
上面两段代码,看似结果一样加了一个final关键字结果却不一样?第一段的s2是字符串b的一个引用,在编译期间无法判断具体的内容,也就是先把引用拿过来,不管内容是什么,最后在运行期间发现是b,虚拟机再对其进行+操作,至于加了一个final关键字,final关键字在编译期间就可以判断他的内容是一个字符串,既然是一个字符串那就要放到常量池中,所以再进行+操作结果为true
String的本质是字符序列,它是通过字符数组实现的!

源码分析:String类中定义了一个char类型的数组,名字为value
接着向下看,String提供了好多创建String对象的方法,通过参数的不同实现构造方法的重载,可以通过byte数组,char数组,int数组,直接给定字符串都可以创建String对象,但是不管这些方法中的参数是什么,最终都是对value数组进行操作。

如果是默认的无参构造方法,创建0个长度的字符数组
看几个源码的例子,虽然每个构造方法参数不同,最终都是对value数组赋值



下面看看String创建对象的方式:
String可以通过直接赋值创建,也可以通过new创建,这两种方式有很大的区别:
总结:对于String str="a"这样的直接赋值字符串常量操作,如果内容相同,Java会认为他们代表的是同一个对象;而用关键字new调用构造方法创建的对象,无论内容是否相同,总是会创建新的对象。
        String s1 = new String("a");
String s2 = new String("a");
System.out.println(s1==s2); String s3 = "v";
String s4 = "v";
System.out.println(s3==s4);

上面代码依次输出false和true,也就验证了上面的话。

验证通过new创建的字符串会放在常量池中吗?

        String s1 = "good";
String s2 = new String("good");
System.out.println(s1==s2); String s3 = "go";
String s4 = s3 + new String("od");
System.out.println(s4 == s1);
System.out.println(s2 == s4);
上面依次输出false,false,false。为什么呢?记住用new关键字。创建的对象总是会在堆区,在堆区有一块空间存这个对象,这个s2就是指向堆区的这块空间,它们有自己的地址空间,使用new创建的对象都是在堆区,而常量池是在方法区的。上面的直接赋值操作,是在编译期就可以确定的,而通过new创建的对象是在程序运行期间才确定。程序运行的String s2=new String("good")的时候,从前向后的顺序,先在栈区创建一个引用s2,向后运行发现有个new,那就在堆区创建一个对象吧,在把s2指向堆区这个对象。但是程序还是是会去常量池中创建一个对象就是这个字符串常量,不过会先进行判断常量池中有没有这个字符串,发现已经有了那么堆区的这个对象就直接指向他。
        String s1 = "ab";
String s2 = "a" + "b";
System.out.println(s1==s2);

上面的输出:true

总结:既然上面的代码输出true说明s2和s1都是指向一个地方,常量池中ab的地址,因为在百衲衣期间,虚拟机就对+进行操作,s2在编译期间就已经确定为ab。
遇到一个问题:为什么输出的String对象是String对象的内容而不是String对象的地址值,我们在平时创建对象输出的时候输出的总是该对象的地址值
        String str = new String("abc");
System.out.println(str);

上面的输出:abc,为什么不是输出str的地址值?因为String类中重写了toString方法,至于平时我们输出的地址值是调用的Object类中的toString方法


创建Integer对象,int的包装类输出也是对象中的值,也是因为Integer类中重写了toString方法。


String类详解的更多相关文章

  1. Java String类详解

    Java String类详解 Java字符串类(java.lang.String)是Java中使用最多的类,也是最为特殊的一个类,很多时候,我们对它既熟悉又陌生. 类结构: public final ...

  2. java 复习整理(四 String类详解)

    String 类详解   StringBuilder与StringBuffer的功能基本相同,不同之处在于StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此效率上S ...

  3. STL之string类详解

    通过在网站上的资料搜集,得到了很多关于string类用法的文档,通过对这些资料的整理和加入一些自己的代码,就得出了一份比较完整的关于string类函数有哪些和怎样用的文档了!下面先罗列出string类 ...

  4. Java的String类详解

    Java的String类 String类是除了Java的基本类型之外用的最多的类, 甚至用的比基本类型还多. 同样jdk中对Java类也有很多的优化 类的定义 public final class S ...

  5. String类详解(1)

    首先String是一个类. 1,实例化String类方法. 1)直接赋值:String name="haha"; 2)通过关键字:String name=new String(&q ...

  6. String类详解,StringBuffer

    先说一下String类的equals()方法. 下面我们先看一段代码: 这段代码输出的结果为: ture true -------------- false 咋看之下貌似Object类比较特别,那么我 ...

  7. Java常用类(一)String类详解

    前言 在我们开发中经常会用到很多的常用的工具类,这里做一个总结.他们有很多的方法都是我们经常要用到的.所以我们一定要把它好好的掌握起来! 一.String简介 1.1.String(字符串常量)概述 ...

  8. 深入理解String类详解

    1.Stringstr = "eee" 和String str = new String("eee")的区别 先看一小段代码, 1 public static ...

  9. C++ string 类详解

    字符串是存储在内存的连续字节中的一系列字符.C++ 处理字符串的方式有两种,一种来自 C 语言,常被称为 C-风格字符串,另一种是基于 string 类库的字符串处理方式.C 风格字符串的处理可以参考 ...

随机推荐

  1. Spark第一个应用程序

    首先要对源码进行编译,生成对应hadoop版本的spark开发程序jar包,上篇已经写了具体的过程,这里不再赘述. 在安装spark的机器上,下载eclipse-java-x86_64版本,将spar ...

  2. Linux传统Huge Pages与Transparent Huge Pages再次学习总结

      Linux下的大页分为两种类型:标准大页(Huge Pages)和透明大页(Transparent Huge Pages).Huge Pages有时候也翻译成大页/标准大页/传统大页,它们都是Hu ...

  3. 经典案例之MouseJack

    引言:在昨天的文章<无线键鼠监听与劫持>中,我们提到今天会向您介绍一个无线键鼠的监听与劫持的经典案例,<MouseJack>:MouseJack能利用无线鼠标和键盘存在的一些问 ...

  4. 规模数据导入高效方式︱将数据快速读入R—readr和readxl包

    本文由雪晴数据网负责翻译整理,原文请参考New packages for reading data into R - fast作者David Smith.转载请注明原文链接http://www.xue ...

  5. 第II篇PCI Express体系结构概述

    虽然PCI总线取得了巨大的成功,但是随着处理器主频的不断提高,PCI总线提供的带宽愈发显得捉襟见肘.PCI总线也在不断地进行升级,其位宽和频率从最初的32位/33MHz扩展到64位/66MHz,而PC ...

  6. R︱sparkR的安装与使用、函数尝试笔记、一些案例

    本节内容转载于博客: wa2003 spark是一个我迟早要攻克的内容呀~ ------------------------------------- 一.SparkR 1.4.0 的安装及使用 1. ...

  7. python︱Anaconda安装、简介(安装报错问题解决、Jupyter Notebook)

    每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 安装完anaconda,就相当于安装了Pyth ...

  8. Windows2003查看远程桌面连接的用户

    要查看通过远程连接windows2003的用户,则打开任务管理器,切换到“用户”选项卡上进行查看.

  9. jQuery提示parsererror错误解决办法

    jquery来处理ajax,用到了json.但是很诧异,jquery的ajax回调时一直调用了error函数(一直提示parsererror异常),success函数一次没执行过 $.ajax({ t ...

  10. MyEclipse10中配置WebLogic10

    MyEclipse10中配置WebLogic10 1.双击打开MyEclipse10,依次操作"Window--->Preferences" 2.在左侧菜单中找到" ...