JavaSE学习笔记(十三)—— StringBuffer

一、StringBuffer概述

  String和StringBuffer他们都可以存储和操作字符串,即包含多个字符的字符串数据。不同的是String类是字符串常量,是不可更改的常量,这时因为String底层使用了一个不可变的字符数组(final char[])。

  

  而StringBuffer是字符串变量,它的对象是可以扩充和修改的,因为它底层使用的是可变的字符数组(char[] value;)

  由于StringBuffer的内部实现方式和String不同,所以StringBuffer在进行字符串处理时,不生成新的对象,在内存使用上要优于String类。所以在实际使用时,如果经常需要对一个字符串进行修改,例如插入、删除等操作使用StringBuffer要更加适合一些。  

  在StringBuffer类中存在很多和String类一样的方法,这些方法在功能上和String类中的功能是完全一样的。但是有一个最显著的区别在于,对于StringBuffer对象的每次修改都会改变对象自身,这点是和String类最大的区别。

  另外由于StringBuffer是线程安全的,关于线程的概念后续有专门的章节进行介绍,所以在多线程程序中也可以很方便的进行使用,但是程序的执行效率相对来说就要稍微慢一些。

二、StringBuffer的构造方法

  • public StringBuffer():无参构造方法,其初始容量为 16 个字符。
  • public StringBuffer(int capacity):指定容量的字符串缓冲区对象
  • public StringBuffer(String str):指定字符串内容的字符串缓冲区对象
public class StringBufferDemo {
    public static void main(String[] args) {
        // public StringBuffer():无参构造方法
        StringBuffer sb = new StringBuffer();
        System.out.println("sb:" + sb);
        // public int capacity():返回当前容量。    理论值
        System.out.println("sb.capacity():" + sb.capacity());//16
        // public int length():返回长度(字符数)。 实际值
        System.out.println("sb.length():" + sb.length());//0
        System.out.println("-----------------");

        // public StringBuffer(int capacity):指定容量的字符串缓冲区对象
        StringBuffer sb2 = new StringBuffer(50);
        System.out.println("sb2:"+sb2);
        System.out.println("sb2.capacity:" + sb2.capacity());//50
        System.out.println("sb2.length():" + sb2.length());//0
        System.out.println("---------------");

        // public StringBuffer(String str):指定字符串内容的字符串缓冲区对象
        StringBuffer sb3 = new StringBuffer("hello");
        System.out.println("sb3:" + sb3);//hello
        System.out.println("sb3.capacity:" + sb3.capacity());//21
        System.out.println("sb3.length:" + sb3.length());//5
    }
}

三、StringBuffer的添加功能

  • public StringBuffer append(String str):可以把任意类型数据添加到字符串缓冲区里面,并返回字符串缓冲区本身
  • public StringBuffer insert(int offset,String str):在指定位置把任意类型的数据插入到字符串缓冲区里面,并返回字符串缓冲区本身
public class StringBufferDemo {
    public static void main(String[] args) {
        // 创建字符串缓冲区对象
        StringBuffer sb = new StringBuffer();
        // public StringBuffer append(String str)
        // StringBuffer sb2 = sb.append("hello");
        // System.out.println("sb:" + sb);//hello
        // System.out.println("sb2:" + sb2);//hello
        // StringBuffer在进行字符串处理时,不生成新的对象
        // System.out.println(sb == sb2);//true

        // 一步一步的添加数据
        // sb.append("hello");
        // sb.append(true);
        // sb.append(12);
        // sb.append(34.56);

        // 链式编程
        sb.append("hello").append(true).append(12).append(34.56);
        System.out.println("sb:" + sb);//hellotrue1234.56

        // public StringBuffer insert(int offset,String str):
        // 在指定位置把任意类型的数据插入到字符串缓冲区里面,并返回字符串缓冲区本身
        sb.insert(5, "word");
        System.out.println("sb:" + sb);//hellowordtrue1234.56
    }
}

四、StringBuffer的删除功能

  • public StringBuffer deleteCharAt(int index):删除指定位置的字符,并返回本身
  • public StringBuffer delete(int start,int end):删除从指定位置开始指定位置结束的内容,并返回本身
public class StringBufferDemo {
    public static void main(String[] args) {
        // 创建对象
        StringBuffer sb = new StringBuffer();

        // 添加功能
        sb.append("hello").append("word").append("java");
        System.out.println("sb:" + sb);

        // public StringBuffer deleteCharAt(int index):删除指定位置的字符,并返回本身
        // 需求:我要删除e这个字符,肿么办?
        // sb.deleteCharAt(1);

        // public StringBuffer delete(int start,int end):删除从指定位置开始指定位置结束的内容,并返回本身
        // 需求:我要删除world这个字符串,肿么办?
        // sb.delete(5, 10);

        // 需求:我要删除所有的数据
        sb.delete(0, sb.length());

        System.out.println("sb:" + sb);
    }
}

五、StringBuffer的替换功能

  public StringBuffer replace(int start,int end,String str):从start开始到end用str替换

public class StringBufferDemo {
    public static void main(String[] args) {
        // 创建字符串缓冲区对象
        StringBuffer sb = new StringBuffer();

        // 添加数据
        sb.append("hello").append("word").append("java");
        System.out.println("sb:" + sb);

        // public StringBuffer replace(int start,int end,String str):从start开始到end用str替换
        // 需求:我要把world这个数据替换为"节日快乐"
        sb.replace(5, 10, "节日快乐");
        System.out.println("sb:" + sb);
    }
}

六、StringBuffer的反转功能

  public StringBuffer reverse():将此字符序列用其反转形式取代

public class StringBufferDemo {
    public static void main(String[] args) {
        // 创建字符串缓冲区对象
        StringBuffer sb = new StringBuffer();

        // 添加数据
        sb.append("霞青林爱我");
        System.out.println("sb:" + sb);

        // public StringBuffer reverse()
        sb.reverse();
        System.out.println("sb:" + sb);
    }
}

七、StringBuffer的截取功能

  • public String substring(int start):返回一个新的 String,它包含此字符序列当前所包含的字符子序列。
  • public String substring(int start,int end):返回一个新的 String,它包含此序列当前所包含的字符子序列。

  注意:返回值类型不再是StringBuffer本身了

public class StringBufferDemo {
    public static void main(String[] args) {
        // 创建字符串缓冲区对象
        StringBuffer sb = new StringBuffer();

        // 添加数据
        sb.append("hello").append("world").append("java");
        System.out.println("sb:" + sb);//helloworldjava

        // 截取功能
        // public String substring(int start)
        String s = sb.substring(5);
        System.out.println("s:" + s);//worldjava
        System.out.println("sb:" + sb);//helloworldjava

        // public String substring(int start,int end)
        String ss = sb.substring(5, 10);
        System.out.println("ss:" + ss);//world
        System.out.println("sb:" + sb);//helloworldjava
    }
}

八、String和StringBuffer的相互转换

  为什么我们要讲解类之间的转换?

  A -- B的转换:我们把A转换为B,其实是为了使用B的功能。

  B -- A的转换:我们把B重新转回来A,因为我们可能要的结果是A类型,所以还得转回来。

public class StringBufferTest {
    public static void main(String[] args) {
        // String → StringBuffer
        String s = "hello";
        // 注意:不能把字符串的值直接赋值给StringBuffer
        // StringBuffer sb = "hello";
        // StringBuffer sb = s;

        // 方式1:通过构造方法
        StringBuffer sb = new StringBuffer(s);

        // 方式2:通过append()方法
        StringBuffer sb2 = new StringBuffer();
        sb2.append(s);
        System.out.println("sb:" + sb);//hello
        System.out.println("sb2:" + sb2);//hello
        System.out.println("-----------");

        // StringBuffer → String
        StringBuffer buffer = new StringBuffer("java");
        // String(StringBuffer buffer)
        // 方式1:通过构造方法
        String str = new String(buffer);

        // 方式2:通过toString()方法
        String str2 = buffer.toString();
        System.out.println("str:" + str);//java
        System.out.println("str:" + str2);//java
    }
}

【练习】

  把数组拼接成一个字符串

public class StringBufferTest2 {
    public static void main(String[] args) {
        // 定义一个数组
        int[] arr = { 44, 33, 55, 11, 22 };

        // 定义功能
        // 方式1:用String做拼接的方式
        String s1 = arrayToString(arr);
        System.out.println(s1);

        // 方式2:用StringBuffer做拼接的方式
        String s2 = arrayToString2(arr);
        System.out.println(s2);
    }

    /**
     * 用StringBuffer做拼接的方式
     * @param arr
     * @return
     */
    public static String arrayToString2(int[] arr) {
        StringBuffer sb = new StringBuffer();
        sb.append("[");
        for (int x = 0; x < arr.length; x++) {
            if (x == arr.length - 1) {
                sb.append(arr[x]);
            } else {
                sb.append(arr[x]).append(",");
            }
        }
        sb.append("]");
        return sb.toString();
    }


    /**
     * 用String做拼接的方式
     * 这种方式会浪费很多常量字符串值,效率会很低
     * @param arr
     * @return
     */
    public static String arrayToString(int[] arr) {
        String s = "";
        s += "[";
        for (int x = 0; x < arr.length; x++) {
            if (x == arr.length - 1) {
                s += arr[x];
            } else {
                s += arr[x];
                s += ",";
            }
        }
        s += "]";
        return s;
    }
}

  把字符串反转

public class StringBufferTest3 {
    public static void main(String[] args) {
        // 键盘录入数据
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入数据:");
        String s = sc.nextLine();

        // 方式1:用String做拼接
        String s1 = myReverse(s);
        System.out.println("s1:" + s1);

        // 方式2:用StringBuffer的reverse()功能
        String s2 = myReverse2(s);
        System.out.println("s2:" + s2);

    }

    /**
     * 用StringBuffer的reverse()功能
     * @param s
     * @return
     */
    public static String myReverse2(String s) {
        return new StringBuffer(s).reverse().toString();
    }

    /**
     * 用String做拼接
     * @param s
     * @return
     */
    public static String myReverse(String s) {
        String result = "";

        char[] chs = s.toCharArray();
        for (int x = chs.length - 1; x >= 0; x--) {
            result += chs[x];
        }

        return result;
    }
}

  判断一个字符串是否是对称字符串。例如"abc"不是对称字符串,"aba"、"abba"、"aaa"、"mnanm"是对称字符串

public class StringBufferTest4 {
    public static void main(String[] args) {
        // 创建键盘录入对象
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入一个字符串:");
        String s = sc.nextLine();

        // 一个一个的比较
        boolean b = isSame(s);
        System.out.println("b:" + b);

        //用字符串缓冲区的反转功能
        boolean b2 = isSame2(s);
        System.out.println("b2:" + b2);
    }

    public static boolean isSame2(String s) {
        return new StringBuffer(s).reverse().toString().equals(s);
    }

    public static boolean isSame(String s) {
        boolean flag = true;

        // 把字符串转成字符数组
        char[] chs = s.toCharArray();
        for (int start = 0, end = chs.length - 1; start <= end; start++, end--) {
            if (chs[start] != chs[end]) {
                flag = false;
                break;
            }
        }
        return flag;
    }
}

九、常见问题

【String、StringBuffer、StringBuilder的区别】

  String是内容不可变的,而StringBuffer,StringBuilder都是内容可变的。

  StringBuffer是同步的,数据安全,效率低;StringBuilder是不同步的,数据不安全,效率高

【StringBuffer和数组的区别】

  二者都可以看出是一个容器,装其他的数据。但是呢,StringBuffer的数据最终是一个字符串数据。而数组可以放置多种数据,但必须是同一种数据类型的。

【String和StringBuffer作形式参数】

  基本类型:形式参数的改变不影响实际参数
  引用类型:形式参数的改变直接影响实际参数

public class StringBufferDemo {
    public static void main(String[] args) {
        String s1 = "hello";
        String s2 = "world";
        System.out.println(s1 + "---" + s2);// hello---world
        change(s1, s2);
        System.out.println(s1 + "---" + s2);// hello---world

        StringBuffer sb1 = new StringBuffer("hello");
        StringBuffer sb2 = new StringBuffer("world");
        System.out.println(sb1 + "---" + sb2);// hello---world
        change(sb1, sb2);
        System.out.println(sb1 + "---" + sb2);// hello---worldworld
    }

    public static void change(StringBuffer sb1, StringBuffer sb2) {
        sb1 = sb2;
        sb2.append(sb1);
    }

    public static void change(String s1, String s2) {
        s1 = s2;
        s2 = s1 + s2;
    }
}

  注意:

  String作为参数传递,效果和基本类型作为参数传递是一样的。因为字符串是常量值。

  StingBuffer作为参数传递时,在方法中直接赋值不改变实际结果,如果是调用方法操作内容时,结果会发生变化。

原文地址:https://www.cnblogs.com/yft-javaNotes/p/10849433.html