java stringbuffer 能存多少g_【技术干货】Java 面试宝典:Java 基础部分(4)

本文详细解析了Java中的String和StringBuffer类。String是不可变类,多次修改会导致内存开销,适合不变的字符串;StringBuffer可变,适用于字符串频繁修改的情况。此外,文中还探讨了String的内存分配、对象创建数量以及String和StringBuffer的区别,强调在性能考虑时选择StringBuffer。同时,文章提到了StringBuffer未覆盖equals方法,可能导致在集合中使用时的问题。
摘要由CSDN通过智能技术生成

海牛学院的 | 第 630 

本文预计阅读 |   12  分钟 466367ad8ca14ca6b7e9f02849fab934.png

85d7656b438979cade3789de8d2e51d8.gif

Java 基础部分

基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语法,集合的语法,io 的语法,虚拟机方面的语法。

31、String s = "Hello";s = s + " world!";这两行代码执行后,原始的 String 对象中的内容到底变了没有?

没有。因为 String 被设计成不可变(immutable)类,所以它的所有对象都是不可变对象。在这段代码中,s 原先指向一个 String 对象,内容是 "Hello",然后我们对 s 进行了+操作,那么 s 所指向的那个对象是否发生了改变呢?答案是没有。这时,s 不指向原来那个对象了,而指向了另一个 String 对象,内容为"Hello world!",原来那个对象还存在于内存之中,只是 s 这个引用变量不再指向它了。

通过上面的说明,我们很容易导出另一个结论,如果经常对字符串进行各种各样的修改,或者说,不可预见的修改,那么使用 String 来代表字符串的话会引起很大的内存开销。因为String 对象建立之后不能再改变,所以对于每一个不同的字符串,都需要一个 String 对象来表示。这时,应该考虑使用 StringBuffer 类,它允许修改,而不是每个不同的字符串都要生成一个新的对象。并且,这两种类的对象转换十分容易。

同时,我们还可以知道,如果要使用内容相同的字符串,不必每次都 new 一个 String。例如我们要在构造器中对一个名叫 s 的 String 引用变量进行初始化,把它设置为初始值,应当这样做:

public class Demo {private String s;...public Demo {s = "Initial Value";}...}

而非

s = new String("Initial Value");

后者每次都会调用构造器,生成新对象,性能低下且内存开销大,并且没有意义,因为 String对象不可改变,所以对于内容相同的字符串,只要一个 String 对象来表示就可以了。也就说,多次调用上面的构造器创建多个对象,他们的 String 类型属性 s 都指向同一个对象。上面的结论还基于这样一个事实:对于字符串常量,如果内容相同,Java 认为它们代表同一个 String 对象。而用关键字 new 调用构造器,总是会创建一个新的对象,无论内容是否相同。

至于为什么要把 String 类设计成不可变类,是它的用途决定的。其实不只 String,很多 Java标准类库中的类都是不可变的。在开发一个系统的时候,我们有时候也需要设计不可变类,来传递一组相关的值,这也是面向对象思想的体现。不可变类有一些优点,比如因为它的对象是只读的,所以多线程并发访问也不会有任何问题。当然也有一些缺点,比如每个不同的状态都要一个对象来代表,可能会造成性能上的问题。所以 Java 标准类库还提供了一个可变版本,即 StringBuffer。

32、是否可以继承 String 类? 

String 类是 final 类故不可以继承。

33、String s = new String("xyz");创建了几个 String Object? 二者之间有什么区别?

两个或一个,”xyz”对应一个对象,这个对象放在字符串常量缓冲区,常量”xyz”不管出现多少遍,都是缓冲区中的那一个。New String 每写一遍,就创建一个新的对象,它一句那个常量”xyz”对象的内容来创建出一个新 String 对象。如果以前就用过’xyz’,这句代表就不会创建”xyz”自己了,直接从缓冲区拿。

34、String 和 StringBuffer 的区别是什么?

JAVA 平台提供了两个类:String 和 StringBuffer,它们可以储存和操作字符串,即包含多个字符的字符数据。这个 String 类提供了数值不可改变的字符串。而这个 StringBuffer 类提供的字符串进行修改。当你知道字符数据要改变的时候你就可以使用 StringBuffer。典型地,你可以使用 StringBuffers 来动态构造字符数据。另外,String 实现了 equals 方法,new String(“abc”).equals(new String(“abc”)的结果为 true,而 StringBuffer 没有实现 equals 方法,所以,new StringBuffer(“abc”).equals(new StringBuffer(“abc”)的结果为 false。

接着要举一个具体的例子来说明,我们要把 1 到 100 的所有数字拼起来,组成一个串。

StringBuffer sbf = new StringBuffer();for(int i=0;i<100;i++){sbf.append(i);}

上面的代码效率很高,因为只创建了一个 StringBuffer 对象,而下面的代码效率很低,因为创建了 101 个对象。

String str = new String();for(int i=0;i<100;i++){    str = str + i;}

在讲两者区别时,应把循环的次数搞成 10000,然后用 endTime-beginTime 来比较两者执行的时间差异,最后还要讲讲 StringBuilder 与 StringBuffer 的区别。String 覆盖了 equals 方法和 hashCode 方法,而 StringBuffer 没有覆盖 equals 方法和 hashCode方法,所以,将 StringBuffer 对象存储进 Java 集合类中时会出现问题。

35、如何把一段逗号分割的字符串转换成一个数组?

如果不查 jdk api,我很难写出来!我可以说说我的思路:

1. 用正则表达式,代码大概为:

String [] result = orgStr.split(“,”);

2. 用 StingTokenizer ,代码为:

StringTokenizer tokener = StringTokenizer(orgStr,”,”);String [] result = new String[tokener .countTokens()];Int i=0;while(tokener.hasNext(){result[i++]=toker.nextToken();}

36、数组有没有 length()这个方法? String 有没有 length()这个方法?

数组没有 length()这个方法,有 length 的属性。

String 有有 length()这个方法。

37 、下面这条语句一共创建了多少个对象:String s="a"+"b"+"c"+"d";

答:对于如下代码:

String s1 = "a";String s2 = s1 + "b";String s3 = "a" + "b";System.out.println(s2 == "ab");System.out.println(s3 == "ab");

第一条语句打印的结果为 false,第二条语句打印的结果为 true,这说明 javac 编译可以对字符串常量直接相加的表达式进行优化,不必要等到运行期去进行加法运算处理,而是在编译时去掉其中的加号,直接将其编译成一个这些常量相连的结果。题目中的第一行代码被编译器在编译时优化后,相当于直接定义了一个”abcd”的字符串,所以,上面的代码应该只创建了一个 String 对象。写如下两行代码,

String s = "a" + "b" + "c" + "d";System.out.println(s == "abcd");

最终打印的结果应该为 true。

38、try {}里有一个 return 语句,那么紧跟在这个 try 后的finally {}里的code会不会被执行,什么时候被执行,在return前还是后? 

也许你的答案是在 return 之前,但往更细地说,我的答案是在 return 中间执行,请看下面程序代码的运行结果:

public class Test {    /**     * @param args add by zxx ,Dec 9, 2008     */    public static void main(String[] args) {        // TODO Auto-generated method stub        System.out.println(new Test().test());;    }        static int test()    {        int x = 1;        try        {            return x;        }        finally        {            ++x;         }    }    }---------执行结果 ---------1

运行结果是 1,为什么呢?主函数调用子函数并得到结果的过程,好比主函数准备一个空罐子,当子函数要返回结果时,先把结果放在罐子里,然后再将程序逻辑返回到主函数。所谓返回,就是子函数说,我不运行了,你主函数继续运行吧,这没什么结果可言,结果是在说这话之前放进罐子里的。

39、下面的程序代码输出的结果是多少?

public class smallT{    public static void main(String args[])    {        smallT t = new smallT();        int b = t.get();        System.out.println(b);    }        public int get()    {        try        {            return 1 ;         }        finally        {            return 2 ;         }     } }返回的结果是 2。

我可以通过下面一个例子程序来帮助我解释这个答案,从下面例子的运行结果中可以发现,try 中的 return 语句调用的函数先于 finally 中调用的函数执行,也就是说 return 语句先执行,finally 语句后执行,所以,返回的结果是 2。Return 并不是让函数马上返回,而是 return 语句执行后,将把返回结果放置进函数栈中,此时函数并不是马上返回,它要执行 finally 语句后才真正开始返回。

在讲解答案时可以用下面的程序来帮助分析:

public class Test {      /**     * @param args add by zxx ,Dec 9, 2008     */    public static void main(String[] args) {        // TODO Auto-generated method stub        System.out.println(new Test().test());;    }        int test(){    try    {        return func1();    }    finally    {        return func2();    } }    int func1()    {        System.out.println("func1");        return 1;    }    int func2()    {        System.out.println("func2");        return 2;    } }-----------执行结果-----------------func1func22

结论:finally 中的代码比 return 和 break 语句后执行。

40、final, finally, finalize 的区别是什么?

final 用于声明属性,方法和类,分别表示属性不可变,方法不可覆盖,类不可继承。内部类要访问局部变量,局部变量必须定义成 final 类型,例如,一段代码……

finally 是异常处理语句结构的一部分,表示总是执行。

finalize 是 Object 类的一个方法,在垃圾收集器执行的时候会调用被回收对象的此方法,可以覆盖此方法提供垃圾收集时的其他资源回收,例如关闭文件等。JVM 不保证此方法总被调用。

4d534e7f3871693779318bd489610aa5.png

总结 以上就是今天的内容了,如果对你有所帮助,希望你能够关注、点赞、转发一键三连支持一下。 需要完整学习线路和配套课堂笔记,请后台回复【学习线路、笔记、思维导图】,更多技术干货请在后台回复【技术论坛】

备注 “ 进群 ” (广告勿扰)

【海牛大数据交流群】

fa150d20f6c13c194b770822a0706678.png 8090655d4fec12759958b8a3ffa0739e.png 4bdf8d8b2c9065887dd5bb65281598f8.png 1f917b8be7ed816c75f1c8c9fa3a37a4.png f89772a3775b5f16b7c0bb6fb5f94804.png 7e4687df649963b8e1c2305fcd8364d0.png  要点赞,要在看 ↓↓↓
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值