zoukankan      html  css  js  c++  java
  • 不可变字符串String与可变字符串StringBuilder、StringBuffer使用详解

    String字符串

    char类型只能表示一个字符,而String可以表示字符串,也就是一个字符序列。但String不是基本类型,而是一个定义好的类,是一个引用类型。在Java中,可以将字符串直接量赋给String类型变量,也可以采用new String(parameter)的形式来创建字符串。

    String str = "This is String”; //用字符串直接量赋值
    String str = new String("Create a String through the Constructor"); //String的构造器来创建字符串

    这里表示引用变量str,引用一个内容为This is String 的字符串对象。 但通常情况下,可直接称str变量是一个字符串,无须过于强调细节。

    字符串特点:String类是不可变类,并且被final修饰,无法继承。 字符串对象都是不可变对象,所以对字符串进行操作时,都是返回新的字符串对象,原有字符串不会改变。

    字符串在程序设计中用得非常频繁,所以掌握String字符串的常用方法是很有必要的。例如:

    String普通方法

    length()        //返回字符串中的字符数,即字符串长度。
    charAt(int index) //根据索引位置,返回对应字符。
    toLowerCase()    //把字符串中所有字母字符变成小写。
    toUpperCase()     //把字符串中所有字母字符变成大写。
    toCharArray()     //把字符串转成字符数组,即一个字符对应一个字符数组元素。
    trim()            //消除字符串的两边空白字符。
            String s = "ABCDEF";
            System.out.println(s.length());  //6
            System.out.println(s.charAt(0)); //索引位置0是 A
            String lowerCase = s.toLowerCase();
            System.out.println(lowerCase); //abcdef
            char[] ch = s.toCharArray();
            System.out.println(ch[0]); // A
            System.out.println("  ABC ".trim()); //ABC

     String比较方法

    equals(Object anObject)           //判断当前字符串对象与指定对象是否相等。
    
    equalsIgnoreCase(String anotherString) //判断当前字符串对象与指定String 对象是否相等,忽略大小写。
    
    compareTo(String anotherString)        //按字典顺序比较两个字符串。
    
    compareToIgnoreCase(String str)     //按字典顺序比较两个字符串,忽略大小写 
    
    boolean startsWith(String prefix)    //判断是否以指定字符串作为前缀
    
    boolean startsWith(String prefix, int toffset) //从指定的索引处开始,判断是否以指定字符串作为前缀
    
    boolean endsWith(String suffix)     //当前字符串是否以指定字符串作为后缀。
    
    contains(CharSequence s)         //如果包含指定字符串,则返回 true。否则false

    注意:比较两个字符串内容是否相等时,很多人都会用 ==操作符来比较,但这是错的,下面说一下它们的区别。

    ==操作符和equals方法的区别

    操作符==比较的是变量值是否相同,因为字符串是引用类型,所以操作符==只能检测两个字符串是否指向同一个引用(对象地址),但无法检测字符串对象的内容。操作符==只能用于判断基本类型以及引用类型是否为null。

    要判断两个字符串变量是否相同,应该使用equals()方法来判断。两个对象指向同一引用,则说明内容相等。否则开始循环比较字符串内容。String的equals()方法实现细节:

        public boolean equals(Object anObject) {
            if (this == anObject) {
                return true;
            }
            if (anObject instanceof String) {
                String anotherString = (String)anObject;
                int n = value.length;
                if (n == anotherString.value.length) {
                    char v1[] = value;
                    char v2[] = anotherString.value;
                    int i = 0;
                    while (n-- != 0) {
                        if (v1[i] != v2[i])
                            return false;
                        i++;
                    }
                    return true;
                }
            }
            return false;
        }
            String s = "This is String";
            String s1 = s + ""; //产生一个新字符串
            System.out.println(s == s1); //两个字符串对象非同一个引用,返回false
            System.out.println(s.equals(s1)); //两个字符串对象的内容相等,返回true

    总结一下就是:操作符==是直接比较变量的值。若是引用类型变量,则比较引用地址是否相同。而equals()方法是比较引用对象中的内容。

    演示其他比较方法

            String s = "hello world";
            System.out.println(s.equalsIgnoreCase("HELLO WORLD")); //忽略大小写,比较字符串内容。true
            //轮流比较字符,字符不相等时,返回字符相减结果:e - a = 4 [69- 65]
            System.out.println(s.compareTo("hallo world")); 
            //判断前缀
            System.out.println(s.startsWith("hello")); //true
            //判断后缀
            System.out.println(s.endsWith("worlds")); // false
            //是否包含指定字符串
            System.out.println(s.contains("lo wor")); //空白字符串也属于字符,别忽略了

    除此之外,还有截取子字符串substring()、切割字符串split()、根据索引找字符串indexOf()、lastIndexOf()以及替换相应字符串replace()等等方法。若要了解更多方法,可查阅API。

    转换字符串 

    有时候,我们需要将某个类型转成字符串来进行操作。例如将数值类型转成字符串,可以很快判断是几位数,可以快速判断是否是回文数。

    方式有两种:采用操作符+拼接成字符串;通过静态方法valueOf(parameter)将相关类型转成对应字符串。

            String s1 = String.valueOf(123);
            String s2 = 123 +"";

    采用操作符+直接转成字符串是最简洁、最方便的。

     字符串直接量与new String()的区别

            String s1 = "ABC";
            String s2 = new String(s1); //等价于 new String("ABC");
            String s3 = s1 + "";
            System.out.println(s1 == s2); //false
            System.out.println(s1 == s3); //false

    字符串直接量是一个字符串对象,而new String(s1)会构建一个同类型但不同内存空间的字符串对象。简单来说就是,两者没有指向同一个引用。所以操作符==判断为false。

    s3 = s1 + ""; 这一段是运行时对字符串变量s1进行拼接,所以会产生一个新的字符串对象。所以为false。

    关于对字符串直接量拼接是否相等同内容字符串

    如果直接对字符串直接量进行拼接,而非采用变量形式拼接字符串,比较结果又会不同。

            String c = "C";
            String s1 = "ABC";
            String s2 = "AB" + c; 
            String s3 = "A" + "B" + "C";
            System.out.println(s1 == s2); //false
            System.out.println(s1 == s3); //true

    因为字符串是常量,所以创建“ABC”这个字符串时,会放在常量池中。 但s2指向的字符串对象引用是到运行时才会确定, 所以会导致创建一个新字符串对象。而s3是以字符串直接量来进行拼接,这些字符串在编译时就能确定下来,编译器就在常量池查找是否存在相同的字符串。若已存在,指向同一个字符串对象,否则,另外创建字符串。

    除了操作符==判断引用地址外,可以使用System.identityHashcode() 查看内存地址是否相同,identityHashcode() 是根据内存地址生成的哈希值。

    可变字符串StringBuilder、StringBuffer

    String是不可变类,字符串都是常量,例如“ABC”会被存储在常量池中。对字符串进行任何更改操作都会产生新的String对象。而StringBuilder与StringBuffer是可变类,它们的字符串对象可以更改,对可变字符串的操作不会生成新的对象,即对同一个字符串对象操作。

    StringBuilder、StringBuffer创建字符串

            StringBuilder sb = new StringBuilder(); //空字符串
            StringBuilder sb1 = new StringBuilder("字符串"); //以String对象作为参数构建字符串

    StringBuilder与StringBuffer构建可变字符串对象都是通过new操作符调用构造方法,不能直接像String一样接收字符串直接量。

    StringBuilder和StringBuffer的使用方式一模一样,两者只需要修改一下类名就可以无缝切换。两者的区别是StringBuilder应用于单线程环境,而StringBuffer应用于多线程环境。所以接下来只以StringBuilder作为示例。 

    可变字符串的常用方法(增删改查)

            append(data)        //将给定数据作为字符串追加到可变字符串尾部
            insert(offset, data)      //将给定数据作为字符串追加到指定的偏移位置
            delete(start, end);       //删除start 到 end -1 的字符
            replace(start, end, str)  //替换start 到 end -1 的字符
            reverse();            //将该字符串进行倒置
            sb.setCharAt(index, ch);; //设置 index 处的字符为给定字符
            StringBuilder sb = new StringBuilder(); //构建空的可变字符串
            sb.append("String is changed");
            System.out.println(sb); //原字符串对象内容被改变 String is changed
            //在下标0处插入123
            sb.insert(0, 123);
            System.out.println(sb); //123String is changed
            //删除 sb字符串的后面7位字符
            sb.delete(sb.length() - 7, sb.length());
            System.out.println(sb); //123String is 
            //替换前三位字符
            sb.replace(0, 3, "!!!");
            System.out.println(sb); //!!!String is 
            //翻转sb字符串
            sb.reverse();
            System.out.println(sb); // si gnirtS!!!
            //设置某个下标的字符内容
            sb.setCharAt(0,'A');
            System.out.println(sb); //Asi gnirtS!!!

    可以看到,对可变字符串作出任何更改操作,都会对其字符串对象进行操作。 相比于String,若是需要大量使用字符串的场景,建议采用可变字符串,可以节省很多内存空间。

    使用可变字符串的场景

    1.将数组内容转成字符串

        public static String toStringWithArray(int[] arr) {
            StringBuilder sb = new StringBuilder("[");
            for(int i = 0; i < arr.length; i++) {
                if(i != arr.length -1) {
                    sb.append(arr[i] + ", ");
                }else {
                    sb.append(arr[i] + "]");
                }
            }
            return sb.toString();
        }

    如果不采用可变字符串,那么在拼接字符串过程中,会产生很多用不到的字符串对象,很浪费内存空间。

    凡是需要大量拼接字符串的地方,都应该尽量使用可变字符串来完成,因为效率高,而且不占内存空间。

  • 相关阅读:
    wc
    1.11考试
    diff
    C++11新利器
    vimdiff
    [学习笔记]多项式
    rev
    [AH2017/HNOI2017]礼物
    tr
    bzoj2555: SubString
  • 原文地址:https://www.cnblogs.com/fwnboke/p/8529453.html
Copyright © 2011-2022 走看看