zoukankan      html  css  js  c++  java
  • String类详解

    看了很多文章事实证明之前的通过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方法。

  • 相关阅读:
    hdu1078 记忆化dfs
    hdu1142 dij+记忆化深搜
    UVA 11374 dijkstra预处理+枚举
    poj1502 单源最短路径
    hdu1814 2-SAT 暴力搜
    macos 10.15.1 pip3安装提示权限不足
    将安装器信息下载到目标卷宗失败
    Mac修改默认python版本
    努力吧,少年
    Implement strStr() 字符串匹配
  • 原文地址:https://www.cnblogs.com/duzhentong/p/7816575.html
Copyright © 2011-2022 走看看