Java—String、StringBuffer、StringBuilder的用法与区别

2024-04-27 12:58

本文主要是介绍Java—String、StringBuffer、StringBuilder的用法与区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文转自:
kingzone_2008的博客


1 String

String:字符串常量,字符串长度不可变。Java中String是immutable(不可变)的。

String类包含如下定义:

/** The value is used for character storage. */  
private final char value[];  /** The offset is the first index of the storage that is used. */  
private final int offset;  /** The count is the number of characters in the String. */  
private final int count;  

用于存放字符的数组被声明为final的,因此只能赋值一次,不可再更改。

2 StringBuffer(JDK1.0)

StringBuffer:字符串变量(Synchronized,即线程安全)。如果要频繁对字符串内容进行修改,出于效率考虑最好使用StringBuffer,如果想转成String类型,可以调用StringBuffer的toString()方法。

Java.lang.StringBuffer线程安全的可变字符序列。在任意时间点上它都包含某种特定的字符序列,但通过某些方法调用可以改变该序列的长度和内容。可将字符串缓冲区安全地用于多个线程。

StringBuffer 上的主要操作是 append 和 insert 方法,可重载这些方法,以接受任意类型的数据。每个方法都能有效地将给定的数据转换成字符串,然后将该字符串的字符追加或插入到字符串缓冲区中。append 方法始终将这些字符添加到缓冲区的末端;而 insert 方法则在指定的点添加字符。例如,如果 z 引用一个当前内容是“start”的字符串缓冲区对象,则此方法调用 z.append(“le”) 会使字符串缓冲区包含“startle”,而 z.insert(4, “le”) 将更改字符串缓冲区,使之包含“starlet”。

3 StringBuilder(JDK5.0)

StringBuilder:字符串变量(非线程安全)。在内部,StringBuilder对象被当作是一个包含字符序列的变长数组。

java.lang.StringBuilder是一个可变的字符序列,是JDK5.0新增的。此类提供一个与 StringBuffer 兼容的 API,但不保证同步。该类被设计用作 StringBuffer 的一个简易替换,用在字符串缓冲区被单个线程使用的时候(这种情况很普遍)。

其构造方法如下:

构造方法描述
StringBuilder()创建一个容量为16的StringBuilder对象(16个空元素)
StringBuilder(CharSequence cs)创建一个包含cs的StringBuilder对象,末尾附加16个空元素
StringBuilder(int initCapacity)创建一个容量为initCapacity的StringBuilder对象
StringBuilder(String s)创建一个包含s的StringBuilder对象,末尾附加16个空元素

在大部分情况下,StringBuilder > StringBuffer。这主要是由于前者不需要考虑线程安全。

4 三者区别

String 类型和StringBuffer的主要性能区别:String是不可变的对象, 因此在每次对String 类型进行改变的时候,都会生成一个新的 String 对象,然后将指针指向新的 String 对象,所以经常改变内容的字符串最好不要用 String ,因为每次生成对象都会对系统性能产生影响,特别当内存中无引用对象多了以后, JVM 的 GC 就会开始工作,性能就会降低。
想象下有如下代码:

public class Test { public static void main(String[] args) { String string = ""; for(int i=0;i<10000;i++){ string += "hello"; } } 
} 

每次对String 类型进行改变的时候,都会生成一个新的 String 对象,也就是说这个循环执行完毕new出了10000个对象,试想一下,如果这些对象没有被回收,会造成多大的内存资源浪费。

使用 StringBuffer 类时,每次都会对 StringBuffer 对象本身进行操作,而不是生成新的对象并改变对象引用。所以多数情况下推荐使用 StringBuffer ,特别是字符串对象经常改变的情况下。

在某些特别情况下, String 对象的字符串拼接其实是被 Java Compiler 编译成了 StringBuffer 对象的拼接,所以这些时候 String 对象的速度并不会比 StringBuffer 对象慢,例如:

String s1 = “我很” + “蓝瘦” + “香菇”;  
StringBuffer Sb = new StringBuilder(“我很”).append(“ simple”).append(“ test”);

生成 String s1对象的速度并不比 StringBuffer慢。其实在Java Compiler里,自动做了如下转换:

String s1 = “我很蓝瘦香菇”;

所以速度很快。但要注意的是,如果拼接的字符串来自另外的String对象的话,Java Compiler就不会自动转换了,速度也就没那么快了,例如:

String s2 = “我很”;  
String s3 = “蓝瘦”;  
String s4 = “香菇”;  
String s1 = s2 + s3 + s4;  

这时候,Java Compiler会规规矩矩的按照原来的方式去做,String的concatenation(即+)操作利用了StringBuilder(或StringBuffer)的append方法实现,此时,对于上述情况,若s2,s3,s4采用String定义,拼接时需要额外创建一个StringBuffer(或StringBuilder),之后将StringBuffer转换为String;若采用StringBuffer(或StringBuilder),则不需额外创建StringBuffer。

5 使用策略

(1)基本原则:如果要操作少量的数据,用String ;单线程操作大量数据,用StringBuilder ;多线程操作大量数据,用StringBuffer。

(2)不要使用String类的”+”来进行频繁的拼接,因为那样的性能极差的,应该使用StringBuffer或StringBuilder类,这在Java的优化上是一条比较重要的原则。例如:

String result = "";  
for (String s : hugeArray) {  result = result + s;  
}  // 使用StringBuilder  
StringBuilder sb = new StringBuilder();  
for (String s : hugeArray) {  sb.append(s);  
}  
String result = sb.toString();  

当出现上面的情况时,显然我们要采用第二种方法,因为第一种方法,每次循环都会创建一个String result用于保存结果,除此之外二者基本相同(对于jdk1.5及之后版本)。

(3)为了获得更好的性能,在构造 StringBuffer 或 StringBuilder 时应尽可能指定它们的容量。当然,如果你操作的字符串长度(length)不超过 16 个字符就不用了,当不指定容量(capacity)时默认构造一个容量为16的对象。不指定容量会显著降低性能。

(4)StringBuilder一般使用在方法内部来完成类似”+”功能,因为是线程不安全的,所以用完以后可以丢弃。StringBuffer主要用在全局变量中。

(5)相同情况下使用 StringBuilder 相比使用 StringBuffer 仅能获得 10%~15% 左右的性能提升,但却要冒多线程不安全的风险。而在现实的模块化编程中,负责某一模块的程序员不一定能清晰地判断该模块是否会放入多线程的环境中运行,因此:除非确定系统的瓶颈是在 StringBuffer 上,并且确定你的模块不会运行在多线程模式下,才可以采用StringBuilder;否则还是用StringBuffer。

参考资料:

http://docs.Oracle.com/javase/tutorial/java/data/buffers.html

http://docs.oracle.com/javase/specs/jls/se7/html/jls-4.html#jls-4.12.4

Java API

6 常见常见的关于String、StringBuffer的面试题
面试题转自:
海子的博客

1.下面这段代码的输出结果是什么?

String a = “hello2”; String b = “hello” + 2;   
System.out.println((a == b));
  
输出结果为:true。原因很简单,”hello”+2在编译期间就已经被优化成”hello2”,因此在运行期间,变量a和变量b指向的是同一个对象。

2.下面这段代码的输出结果是什么?

  String a = “hello2”;   String b = “hello”; String c = b + 2; System.out.println((a == c));

  输出结果为:false。由于有符号引用的存在,所以 String c = b + 2;不会在编译期间被优化,不会把b+2当做字面常量来处理的,因此这种方式生成的对象事实上是保存在堆上的。因此a和c指向的并不是同一个对象。

3.下面这段代码的输出结果是什么?

  String a = “hello2”;   final String b = “hello”; String c = b + 2; System.out.println((a == c));

  输出结果为:true。对于被final修饰的变量,会在class文件常量池中保存一个副本,也就是说不会通过连接而进行访问,对final变量的访问在编译期间都会直接被替代为真实的值。那么String c = b + 2;在编译期间就会被优化成:String c = “hello” + 2;

4.下面这段代码输出结果为:

public class Test { public static void main(String[] args) { String a = "hello2";  final String b = getHello(); String c = b + 2;  System.out.println((a == c)); } public static String getHello() { return "hello"; } 
} 

输出结果为false。这里面虽然将b用final修饰了,但是由于其赋值是通过方法调用返回的,那么它的值只能在运行期间确定,因此a和c指向的不是同一个对象。

5.下面这段代码的输出结果是什么?

public class Main { public static void main(String[] args) { String a = "hello"; String b =  new String("hello"); String c =  new String("hello"); String d = b.intern(); System.out.println(a==b); System.out.println(b==c); System.out.println(b==d); System.out.println(a==d); } 
} 
false
false
false
true

这里面涉及到的是String.intern方法的使用。在String类中,intern方法是一个本地方法,在JAVA SE6之前,intern方法会在运行时在常量池中查找是否存在内容相同的字符串,如果存在则返回指向该字符串的引用,如果不存在,则会将该字符串入池,并返回一个指向该字符串的引用。因此,a和d指向的是同一个对象。

6.String str = new String(“abc”)创建了多少个对象?

这个问题在很多书籍上都有说到比如《Java程序员面试宝典》,包括很多国内大公司笔试面试题都会遇到,大部分网上流传的以及一些面试书籍上都说是2个对象,这种说法是片面的。

这段代码在运行期间只创建了一个对象,即在堆上创建了”abc”对象。而为什么大家都在说是2个对象呢,这里面要澄清一个概念 该段代码执行过程和类的加载过程是有区别的。在类加载的过程中,确实在运行时常量池中创建了一个”abc”对象,而在代码执行过程中确实只创建了一个String对象。

  因此,这个问题如果换成 String str = new String(“abc”)涉及到几个String对象?合理的解释是2个。

  个人觉得在面试的时候如果遇到这个问题,可以向面试官询问清楚”是这段代码执行过程中创建了多少个对象还是涉及到多少个对象“再根据具体的来进行回答。

7.下面这段代码1)和2)的区别是什么?

public class Test { public static void main(String[] args) { String str1 = "I"; //str1 += "love"+"java";        1) str1 = str1+"love"+"java";      //2) } 
} 

 1)的效率比2)的效率要高,1)中的”love”+”java”在编译期间会被优化成”lovejava”,而2)中的不会被优化。在1)中只进行了一次append操作,而在2)中进行了两次append操作。

这篇关于Java—String、StringBuffer、StringBuilder的用法与区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/940569

相关文章

区块链中的APP与传统APP的区别

一、技术 区块链中的APP是基于区块链技术开发的,而传统APP则基于传统的应用程序商店或网页。区块链中的APP利用区块链技术的去中心化、数据不可篡改等特点,使得应用程序的开发和分发更加安全、透明和可信。与传统APP相比,区块链中的APP无需中心化服务器或第三方机构即可运行,降低了开发成本和维护成本。 二、应用场景 传统APP的应用场景较为广泛,涵盖了社交、娱乐、购物、金融等多个领域。而区

微信小程序和微信H5微商城的区别,微信商城主要有哪些类型

什么是微信小程序? 微信小程序的推出是一场革命,它为用户提供了应用程序中的应用程序。这些轻量级的、不到10 MB的实用程序改变了微信的体验,使其从一个纯粹的消息传递平台转变为一个包罗万象的数字宇宙。截至2024年初,WeChat拥有超过430万个小程序,据说超过95%的公司在应用程序中拥有自己的小程序。 微信日活跃用户达4.5亿,微信的使用统计数据显示,平均每人使用9.8个小程序。 微信

margin和padding在开发者模式下的区别

设置margin有粉红色背景 设置padding无粉红色背景

Java中的Mysql数据库备份与定时任务快速实现(详细代码示例)

引言 在现代软件系统中,数据库备份是确保数据安全的关键措施之一。通过定期备份,可以在数据丢失或损坏时迅速恢复,从而减少潜在的业务风险。Java作为一种广泛使用的编程语言,提供了多种实现数据库备份的方法。本文将介绍如何使用Java编写一个定时备份数据库的程序,并详细解释每个步骤的实现细节。 一、开发环境 SpringBoot项目项目中添加MySQL的JDBC驱动依赖 <!-- mysql 驱

[Struts2] page、request、session、application的区别

page page指当前页面,在一个jsp页面中有效request request值从http请求到服务器处理结束,返回响应的整个过程,在这个过程中使用forward方式跳转多个jsp,在这些页面里都可以使用这个变量session 当前会话中有效,从浏览器打开到浏览器关闭这个过程,每个用户都有唯一的一个application 它的有效范围是整个应用,作用域里的变量它们的存活时间是最长的,如果不进行

[SpringMVC] 今日最坑

折腾了一晚上,在将一个Eclipse项目转移到IDEA中时,发现<c:if>标签始终不起作用,到最后才发现问题出在了web.xml文件中,IDEA默认创建的web.xml文件是这样的: <web-app><display-name>Archetype Created Web Application</display-name></web-app> 这样将SpringMVC的环境搭建好后,<c

[SpringMVC] SpringMVC

RequestMapping注解 SpringMVC使用@RequestMapping注解为控制器指定可以处理哪些URL请求在控制器的类定义及方法定义处都可标注 类定义处:提供初步的请求映射信息,相对于WEB应用的根目录方法定义处:提供进一步的细分映射信息,相对于类定义处的URL,若类定义处未标注@RequestMapping,则方法标记处的URL相当于WEB应用的根目录DispatchSer

[Spring] 基于注解来配置Bean

Bean的配置方式 基于XML文件的方式基于注解的方式(基于注解配置Bean,基于注解来装配Bean的属性) 在classpath中扫描组件 组件扫描:Spring能够从classpath下自动扫描、侦测和实例化具有特定注解的组件特定组件包括: @component:基本注解,标识了一个受Spring管理的组件@Respositoy:标识持久层组件@Service:标识服务层组件@Contr

[Spring] Spring配置文件头部xmlns配置

刚开始学习Spring或许都对其配置文件头部的各种配置表示困惑,先贴出一个比较常见的头部文件 <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework

[Spring] Spring JDBCTemplate

JdbcTemplate简介 为了使JDBC更加易于使用,Spring在JDBC API定义了一个抽象层,以此建立一个JDBC存取框架作为Spring JDBC框架的核心,JDBC模板的设计目的是为不同类型的JDBC操作提供模板方法,每个模板方法都能控制整个过程,并允许覆盖过程中的特定任务 JdbcTemplate的使用 加入jar包 c3p0-0.9.1.2.jarcom.springso