String 、StringBuffer、StringBuilder 三者的异同(重点面试题)

简介: String 、StringBuffer、StringBuilder 三者的异同(重点面试题)

String 、StringBuffer、StringBuilder 三者的异同。

一:


字符序列可不可变

String     不可变

StringBuffer     可变

StringBuilder      可变

二:


StringBuffer 和 StringBuilder 的一些区别

StringBuffer 线程安全,效率低。底层为char [] , jdk1.0。

StringBuilder 线程不安全,使用效率高 底层为 char[]  , jdk 5.0 新增。

三 :思考为什么String是不可变的 StringBuffer 和 StringBUilders是可变的?


我们可以通过查看 以上三者的源码来分析。


String:

image.png



StringBuffer:

image.png



StringBuilders:

image.png



我们可以通过上面的三个底层源码可以看到三者都是用char[ ]来存储,为什么就 String 是不可变的呢? 我相信细心的小伙伴已经发现了 ,就 String 有 final ,则String是不可以变的。(如果有小伙伴对 final 不是很了解,我过几天会总结一下final的知识点。)


上面虽然简述了为什么可变不可变的原理,但是面试题可没上面几句简单,所以下面就是正真的重点!!!仔细看!!!


我们先进行源码的分析:


String str = new String(); //char[0]
String str1 = new String("abc");//char[] value = new char[]{'a','b','c'};

我们能知道这个是String的创建方法。


StringBuffer stringBuffer = new StringBuffer();

接下来我们查看一下StringBuffer的底层源码:

image.png



再查看super:

image.png



在查看value:


image.png


从源码我们能知道 StringBuffer 在创建一个新空间时,是首先构造一个字符串生成器,其中不包含字符,初始容量为16个字符。


StringBuffer stringBuffer = new StringBuffer();
//相当于底层创建了一个长度为16 char[] 数组
stringBuffer.append('a'); //value[0] = 'a';
stringBuffer.append('b'); //value[1] = 'b';
stringBuffer.append('c'); // value[2] = 'c';

当我们使用:


StringBuffer stringBuffer = new StringBuffer("abc");

我们查看底层源码:

image.png



我们能知道他是会创建一个 str.length() + 16 的数组长度。


StringBuffer stringBuffer = new StringBuffer("abc");
//char [] value = new char ["abc".length() + 16]

通过上述对源码的查询我们能知道StringBuffer之所以可变是因为他有大量的数组空间,我们可以通过在数组上修改通过对字符串的修改!!!


这样我们就正真的知道了为啥 StringBuffer 是可以变的了(StringBuilders和StringBuffer 的结构是相同的,上述结论也符合StringBuilders)


拓展一:


我们接下来要对StringBuffer的可变进行一些问题的延申:


我们通过上述源码分析我们能知道因为StringBuffer对char[] 数组的增大才可以使字符串变成可变性,一般情况StringBuffer初始话就比字符串长度多16个,如果我们一直对字符串进行添加处理直到超过初始化定义的长度会怎么办呢???


是会溢出还是会有一些解决办法呢???


我们可以在对源码进行分析:

image.png





这段源码的作用就是对添加后的字符串长度是否超过底层数组的长度进行一个判断,如果超过原来的底层数组就会执行newCapacity这个函数。接下来我们再来查看这个函数的源码:

image.png



通过对这个源码的分析我们能知道:这个是创建一个原来长度两倍加+2的长度的数组。


再来一起分析上面的几个代码我们可以知道如果原来的数组容量不够的情况下,我们会创建一个为原来长度两倍加2的一个新数组,再把原来的数组内容复制到现在这个数组中,并且在进行判断添加后的数组是否能大于添加后的长度,如果还是达不到则继续创建新的更大的数组,直到能大于目前添加后字符串的长度为止。



拓展二:


String、StringBuffer 和 StringBuilders三者的效率比较。


我们可以通过下面执行的代码来判断一下三者的执行效率:


package StringMoth;
public class demo3 {
    public static void main(String[] args) {
        long start_time = 0L;
        long end_time = 0L;
        String text = "";
        StringBuffer stringBuffer = new StringBuffer("");
        StringBuilder stringBuilder = new StringBuilder("");
        //开始对比
        start_time = System.currentTimeMillis();
        for (int i = 0; i < 20000; i++) {
            stringBuffer.append(String.valueOf(i));
        }
        end_time = System.currentTimeMillis();
        System.out.println("Buffer的执行时间为:"+(end_time-start_time));
        start_time = System.currentTimeMillis();
        for (int i = 0; i < 20000; i++) {
            stringBuilder.append(String.valueOf(i));
        }
        end_time = System.currentTimeMillis();
        System.out.println("Builder的执行时间为:"+(end_time-start_time));
        start_time = System.currentTimeMillis();
        for (int i = 0; i < 20000; i++) {
            text = text + i;
        }
        end_time = System.currentTimeMillis();
        System.out.println("String的执行时间为:"+(end_time-start_time));
    }
}

输出结果:

image.png



我们能直到StringBuilders的效率是最高的,Buffer其次而String是效率最低的。


我们通过上面介绍的底层原理我们能知道String是不可变性的,如果添加一个新的字符串他则会开辟一个新的空间,然后把新的字符串放进这个新的地址中,因为每一次添加Sting都会开辟一个新的空间,所以会导致String的效率会很慢。


而Buffer和Builders则不同,它们都有多余的数组空间,在字符串长度不大于原数组长度时不会开辟新的空间,而是添加到空余的数组里面,这样就会使其效率高的很多!


而Builders是没有锁的,它可以同时进行多个操作,而Buffer是有锁的,只能同时进行一个操作,导致效率没有Builders的效率高。(这个地方我不是很确定,如果有大佬觉得错了希望能指正一下,我下去我也在查查别的资料!!)



相关文章
|
3月前
|
安全 Java API
【Java字符串操作秘籍】StringBuffer与StringBuilder的终极对决!
【8月更文挑战第25天】在Java中处理字符串时,经常需要修改字符串,但由于`String`对象的不可变性,频繁修改会导致内存浪费和性能下降。为此,Java提供了`StringBuffer`和`StringBuilder`两个类来操作可变字符串序列。`StringBuffer`是线程安全的,适用于多线程环境,但性能略低;`StringBuilder`非线程安全,但在单线程环境中性能更优。两者基本用法相似,通过`append`等方法构建和修改字符串。
58 1
|
2月前
|
安全 Java
String、StringBuffer、StringBuilder的区别
这篇文章讨论了Java中String、StringBuffer和StringBuilder的区别。String是不可变的,每次操作都会产生新的对象,效率低且浪费内存。StringBuilder可以在原字符串基础上进行操作,不开辟额外内存,弥补了String的缺陷。StringBuffer和StringBuilder类似,但StringBuffer的方法是线程安全的。文章还列举了StringBuffer的常用方法,并提供了使用示例代码。最后总结了这三者的主要区别。
String、StringBuffer、StringBuilder的区别
|
27天前
|
canal 安全 索引
(StringBuffer和StringBuilder)以及回文串,字符串经典习题
(StringBuffer和StringBuilder)以及回文串,字符串经典习题
33 5
|
10天前
|
存储 安全 Java
String、StringBuffer 和 StringBuilder 的区别
【10月更文挑战第21天】String、StringBuffer 和 StringBuilder 都有各自的特点和适用场景。了解它们之间的区别,可以帮助我们在编程中更合理地选择和使用这些类,从而提高程序的性能和质量。还可以结合具体的代码示例和实际应用场景,进一步深入分析它们的性能差异和使用技巧,使对它们的理解更加全面和深入。
7 0
|
2月前
|
安全 Java
Java StringBuffer 和 StringBuilder 类详解
在 Java 中,`StringBuffer` 和 `StringBuilder` 用于操作可变字符串,支持拼接、插入、删除等功能。两者的主要区别在于线程安全性和性能:`StringBuffer` 线程安全但较慢,适用于多线程环境;`StringBuilder` 非线程安全但更快,适合单线程环境。选择合适的类取决于具体的应用场景和性能需求。通常,在不需要线程安全的情况下,推荐使用 `StringBuilder` 以获得更好的性能。
|
3月前
|
安全 Java API
Java系类 之 String、StringBuffer和StringBuilder类的区别
这篇文章讨论了Java中`String`、`StringBuffer`和`StringBuilder`三个类的区别,其中`String`是不可变的,而`StringBuffer`是线程安全的可变字符串类,`StringBuilder`是非线程安全的可变字符串类,通常在单线程环境下性能更优。
Java系类 之 String、StringBuffer和StringBuilder类的区别
|
3月前
|
API C# 开发者
WPF图形绘制大师指南:GDI+与Direct2D完美融合,带你玩转高性能图形处理秘籍!
【8月更文挑战第31天】GDI+与Direct2D的结合为WPF图形绘制提供了强大的工具集。通过合理地使用这两种技术,开发者可以创造出性能优异且视觉效果丰富的WPF应用程序。在实际应用中,开发者应根据项目需求和技术背景,权衡利弊,选择最合适的技术方案。
127 0
|
2月前
|
Java 索引
java基础(13)String类
本文介绍了Java中String类的多种操作方法,包括字符串拼接、获取长度、去除空格、替换、截取、分割、比较和查找字符等。
36 0
java基础(13)String类
|
20天前
|
Java
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
本文深入探讨了Java中方法参数的传递机制,包括值传递和引用传递的区别,以及String类对象的不可变性。通过详细讲解和示例代码,帮助读者理解参数传递的内部原理,并掌握在实际编程中正确处理参数传递的方法。关键词:Java, 方法参数传递, 值传递, 引用传递, String不可变性。
41 1
【编程基础知识】(讲解+示例实战)方法参数的传递机制(值传递及地址传递)以及String类的对象的不可变性
|
17天前
|
安全 Java 测试技术
Java零基础-StringBuffer 类详解
【10月更文挑战第9天】Java零基础教学篇,手把手实践教学!
17 2