Java常用类

4.字符串相关类练习

4.1StringBuilder练习

package li.normalclass.stringbuilder;

public class TestBuffer {
    public static void main(String[] args) {
        StringBuffer a = new StringBuffer("A");
        StringBuffer b = new StringBuffer("B");
        mb_operate(a,b);
        System.out.println(a+"."+b);
}

    private static void mb_operate(StringBuffer x, StringBuffer y) {
        x.append(y);
        y=x;
    }
}

问: System.out.println(a+"."+b);的输出结果是?

答:AB.B

原因如下:

当运行main方法时,在栈中开辟了两个内存空间,分别是a和b,指向在堆中创建的两个对象,对象又分别指向了两个字符数组,数组中存储的就是新建对象时传入的参数。

接着运行mb_operate方法,在栈中另外开辟两个内存空间,命名为x和y。其中x指向的是a指向的对象,y指向的是b指向的对象。 x.append(y);就是在a对象指向的数组中追加b对象指向的数组中的值,此时a对象指向的数组值为AB;y=x;就是将x中存储的地址赋值给y,之后y就指向了x指向的对象,即a指向的对象。

System.out.println(a+"."+b);输出a对象和b对象的值,此时a对象中的值为AB,b对象中的值为B,因此输出的内容为AB.B

4.2String和字符数组练习

package li.normalclass.stringbuilder;

public class Example {
    String str = new String("good");
    char[] ch = {'a','b','c'};
    public static void main(String[] args) {
        Example ex = new Example();
        ex.change(ex.str,ex.ch);
        System.out.print(ex.str+"and");
        System.out.print(ex.ch);
    }

    public void change(String str,char ch[]){
        str = "test ok";
        ch[0] = 'g';
    }
}

问:

System.out.print(ex.str+"and");

“System.out.print(ex.ch);`

的输出结果是?

答:goodandgbc

原因如下:

首先,当main方法创建ex对象,ex指向的对象中的str会再创建一个String类型对象,str指向这个String对象,然后String对象又指向good字符串的地址。

ex指向的对象中的ch则会创建一个字节数组,该数组中存储了abc三个字符。

当运行change()方法时,又在栈中开辟了两个局部变量空间,分别为str和ch,然后传入参数。将ex.str指向的地址0x2012赋给局部变量str,将ex.ch指向的地址0x4012赋给局部变量ch。在change方法体中,str = "test ok";则是在常量池中新建了一个字符串,将新建字符串的地址赋给了局部变量str,因此此时局部变量str指向的地址为0x5012;而 ch[0] = 'g';,则是将局部变量ch指向的字符数组中的ch[0]数据改变为g。

此时,运行语句System.out.print(ex.str+"and");则输出的是仍ex.str指向的String对象指向的字符串good

运行语句 System.out.print(ex.ch);,输出的是被局部变量改变的数组gbc,因为ex.ch和局部变量ch指向的地址相同。

4.3再次理解String和StringBuilder的不同之处

4.3.1String

package li.normalclass.stringbuilder;

public class TestStringAndBuilder {
    public static void main(String[] args) {
        String str5 = new String("北京");
        str5 = str5.concat("故宫");
        str5 = str5.concat("博物院");
        System.out.println(str5);//北京故宫博物院
    }
}

采用字面值的方式创建一个字符串时,JVM首先会去字符串池中查找是否存在”北京”这个对象,如果不存在,则在字符串池中创建”北京”这个对象,然后将池中”北京”这个对象的引用地址返回给”北京”对象的引用str5。使用concat()方法可以追加子字符串,但是String是不可变长序列,所以是实际上是在常量池重新创建了一个对象,并把追加的字符串连同原字符串一同赋值给新的对象,然后将新对象的引用地址返回给str5,这样str5就指向了一个新的地址空间。每次使用concat()方法追加子串都会经历上述过程,str5的指向不断改变,最终会指向最后一次开辟的对象地址

因此使用concat()追加子串的方法效率无疑是很低的,那么有没有一种办法可以直接在创建的对象里添加子串呢?这就是我们要涉及到的StringBuilder类

4.3.2StringBuilder

package li.normalclass.stringbuilder;

public class TestStringAndBuilder {
    public static void main(String[] args) {
 
        StringBuilder builder = new StringBuilder("北京");
        builder.append("故宫");
        builder.append("地址为北京市东城区景山前街4号");
        System.out.println(builder.toString());
    }
}

StringBuilder对象指向的字符数组是可变的,在创建时的长度为创建时传入的字符长度+16,之后每次增加字符进去,直到要增加的字符长度超过当前字符数组的内存空间容量,就会重新开辟一块地址空间,容量为当前容量的两倍再加上2,然后将StringBuilder对象指向新开辟的内存地址,而在栈内存中的builder存储的地址保持不变,指向StringBuilder对象。