java字符连接字符串数组_Java中连接字符串的最佳方法

java字符连接字符串数组

最近有人问我这个问题–在Java中使用+运算符连接字符串是否对性能不利?

这让我开始思考Java中连接字符串的不同方法,以及它们如何相互对抗。 这些是我要研究的方法:

  1. 使用+运算符
  2. 使用StringBuilder
  3. 使用StringBuffer
  4. 使用String.concat()
  5. 使用String.joinString.join新增功能)

我还尝试了String.format()但是它是如此缓慢,以至于我暂时不在本文中介绍。

在继续之前,我们应该分离两个用例:

  1. 将两个字符串串联在一起作为一个调用,例如在日志消息中。 由于这只是一个电话,您可能会认为性能几乎不是问题,但结果仍然很有趣,并且可以阐明该主题。
  2. 在一个循环中连接两个字符串。 在这里,性能更是一个问题,尤其是当循环很大时。

我最初的想法和问题如下:

  1. +运算符是用StringBuilder实现的,因此至少在连接两个String的情况下,它应产生与StringBuilder类似的结果。 幕后到底发生了什么?
  2. 在所有类的设计目的都是为了连接Strings并取代StringBuffer之后,StringBuilder应该是最有效的方法。 但是,与String.concat()相比,创建StringBuilder的开销是多少?
  3. StringBuffer是连接字符串的原始类–不幸的是,其方法是同步的。 确实不需要同步,随后它被不同步的StringBuilder代替。 问题是,JIT是否优化了同步?
  4. String.concat()应该适用于2个字符串,但是在循环中是否可以正常工作?
  5. String.join()比StringBuilder具有更多的功能,如果我们指示它使用空的定界符来连接String,它将如何影响性能?

我要解决的第一个问题是+运算符的工作方式。 我一直都知道它在幕后使用了StringBuilder,但是要证明这一点,我们需要检查字节码。

如今 ,查看字节码最简单的方法是使用JITWatch ,这是一个非常出色的工具,旨在了解JIT如何编译您的代码。 它有一个很棒的视图,您可以在其中与字节码(如果要转到该级别,还可以是机器码)并排查看源代码。

屏幕截图2015-02-17 at 17.27.46

这是一个非常简单的方法plus2()的字节码,我们可以看到确实在第6行上创建了一个StringBuilder,并附加了变量a(第14行)和b(第18行)。

我认为将其与StringBuffer的手工使用进行比较会很有趣,因此我创建了另一个方法build2(),结果如下。

屏幕截图2015年2月17日在17.31.37

此处生成的字节码不如plus()方法那么紧凑。 StringBuilder存储在变量高速缓存中(第13行),而不是仅留在堆栈上。 我不知道为什么会这样,但是JIT也许可以做到这一点,我们将不得不看看时机如何。

无论如何,如果用plus运算符和StringBuilder将2个字符串连接在一起的结果显着不同,那将是非常令人惊讶的。

我写了一个小型的JMH测试来确定不同方法的执行方式。 首先让我们看一下两个Strings测试。 参见下面的代码:

package org.sample;import org.openjdk.jmh.annotations.*;
import org.openjdk.jmh.infra.Blackhole;import java.util.UUID;
import java.util.concurrent.TimeUnit;@Warmup(iterations = 5, time = 1, timeUnit = TimeUnit.SECONDS)
@Measurement(iterations = 5, time = 1, timeUnit = TimeUnit.SECONDS)
@Fork(1)
@State(Scope.Thread)
public class LoopStringsBenchmark {private String[] strings;@Setuppublic void setupTest(){strings = new String[100];for(int i = 0; i<100; i++) {strings[i] = UUID.randomUUID().toString().substring(0, 10);}}@Benchmarkpublic void testPlus(Blackhole bh) {String combined = "";for(String s : strings) {combined = combined + s;}bh.consume(combined);}@Benchmarkpublic void testStringBuilder(Blackhole bh) {StringBuilder sb = new StringBuilder();for(String s : strings) {sb.append(s);}bh.consume(sb.toString());}@Benchmarkpublic void testStringBuffer(Blackhole bh) {StringBuffer sb = new StringBuffer();for(String s : strings) {sb.append(s);}bh.consume(sb.toString());}@Benchmarkpublic void testStringJoiner(Blackhole bh) {bh.consume(String.join("", strings));}@Benchmarkpublic void testStringConcat(Blackhole bh) {String combined = "";for(String s : strings) {combined.concat(s);}bh.consume(combined);}
}

结果如下:

屏幕+截图+ 2015-02-17 + at + 17.41.26

显而易见的赢家是String.concat()。 毫不奇怪,因为它不必为每次调用创建StringBuilder / StringBuffer而付出性能损失。 虽然确实需要每次都创建一个新的String(这将在以后变得很重要),但是对于连接两个Sting的非常简单的情况,它更快。

另一点是,尽管产生了额外的字节码,但正如我们预期的那样,plus和StringBuilder是等效的。 StringBuffer仅比StringBuilder慢一点,这很有趣,这表明JIT必须做一些魔术来优化同步。

下一个测试将创建一个100个字符串的数组,每个字符串包含10个字符。 基准测试比较了将100个字符串连接在一起的不同方法所花费的时间。 参见下面的代码:

这次的结果看起来完全不同:

屏幕截图2015-02-17 at 17.54.37

在这里,加号方法确实遭受了损失。 每当您遍历循环时,创建StringBuilder的开销就会减少。 您可以在字节码中清楚地看到这一点:

屏幕截图2015-02-17 at 17.59.46

您可以看到每次执行循环时都会创建一个新的StringBuilder(第30行)。 JIT应该发现这一点并能够对其进行优化是有争议的,但是事实并非如此,使用+变得非常慢。

同样,StringBuilder和StringBuffer的性能完全相同,但是这次它们都比String.concat()快。 String.concat()在循环的每次迭代中创建新的String所付出的代价最终会增加,并且StringBuilder变得更加高效。

给定可以添加到此方法的所有其他功能,String.join()的效果非常好,但是,正如预期的那样,对于纯串联而言,它不是最佳选择。

摘要

如果要在单行代码中连接字符串,则我将使用+运算符,因为它最易读,并且对于单个调用而言,性能实际上并不重要。 还要提防String.concat(),因为您几乎肯定会需要执行空值检查 ,而其他方法则不需要这样做。

在循环中连接字符串时,应使用StringBuilder。 您可以使用StringBuffer,但我不一定在所有情况下都信任JIT来像基准测试中那样高效地优化同步。

我的所有结果都是使用JMH取得的,并且都带有通常的健康警告 。

翻译自: https://www.javacodegeeks.com/2015/02/optimum-method-concatenate-strings-java.html

java字符连接字符串数组

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/338537.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微型计算机使用的普通编码是,2017计算机一级考试选择题练习及答案(2)

35、一般计算机硬件系统的主要组成部件有五大部分&#xff0c;下列选项中不属于这五大部分的事( B )A)运算器 B)软件C)输入设备和输出设备 D)控制器36、微型计算机主机的主要组成部分有( A)。A)运算器和控制器B)CPU 和软盘C)CPU 和显示器D)CPU、内存储器和硬盘37、微型计算机硬…

C 构造函数重载

C 用参数初始化表对数据成员初始化在C 中&#xff0c;构造函数的函数可以通过体内赋值语句对数据成员实现初始化&#xff0c;C 还提供另一种初始化数据成员的方法&#xff0c;用参数初始化表来实现对数据成员的初始化。这种方法不在函数体内对数据成员初始化&#xff0c;而是在…

乐播投延迟很高_大屏也要高刷新!华为4K@120智慧屏初体验,屏幕软件都够硬

买电视就是买屏幕&#xff0c;华为智慧屏也不例外。可参数大而全的旗舰产品往往都有着相当高昂的售价&#xff0c;假如你不愿意将就千元级别的画质&#xff0c;希望电视能坚持更长时间&#xff0c;却又没有足以点满所有技能树的预算……华为新出的S系列智慧屏&#xff0c;大概正…

java常见性能优化_十大最常见的Java性能问题

java常见性能优化Java性能是所有Java应用程序开发人员都关心的问题&#xff0c;因为快速使应用程序与使其正常运行同等重要。 史蒂文海恩斯&#xff08;Steven Haines&#xff09;使用他在Java性能问题上的个人经验得出的结论是&#xff0c; 大多数问题都有共同的根本原因 。 因…

为甚 html 显示为源码,显示字符串作为美化HTML源代码

我正在开发一个ASP.NET MVC Web应用程序。在一个页面上&#xff0c;我有一个字符串&#xff0c;如下所示&#xff1a;This is some text !对于预览模式&#xff0c;我使用Html.Raw(Model.MyText)&#xff0c;文本显示为呈现的HTML&#xff1a;此是一些文字&#xff01;我现在想…

C语言 | 链表概述

C语言链表概述链表是一种常见的重要的数据结构。它是动态地进行存储分配的一种结构&#xff0c;是根据需要开辟内存单元。链表有一个“头指针”变量&#xff0c;它存放一个地址&#xff0c;该地址指向一个元素。链表中每一个元素称为“结点”&#xff0c;每个结点都应包括两个部…

笔记本电脑频繁自动重启_如何解决电脑频繁自动重启

电脑频繁自动重启的原因可能是系统问题&#xff0c;也可能是电脑硬件出现问题&#xff0c;需要经过仔细排查才能找到自动重启的原因。电脑频繁自动重启的原因一、显卡或CPU温度过高当显卡和CPU的温度达到一定温度以后&#xff0c;主板就会启用自动保护功能&#xff0c;使电脑重…

jooq_jOOQ与Hibernate:何时选择哪个

jooqHibernate已成为Java生态系统中的事实上的标准&#xff0c;事实上&#xff0c; 如果标准对您很重要 &#xff0c;并且如果您将JCP与ISO&#xff0c;ANSI&#xff0c;IEEE等置于同一级别&#xff0c;那么Hibernate也是实际的JavaEE标准实现。 本文的目的不是讨论标准&#…

html5编辑器新手用,3款容易上手的HTML5编辑工具推荐~

该楼层疑似违规已被系统折叠 隐藏此楼查看此楼我们的生活已经时刻可见H5页面的身影&#xff0c;从美食到用车、从购物到求职&#xff0c;可以说HTML5技术对我们生活的影响是颠覆性的。今天小编就给大家推荐5款容易上手的H5页面&#xff0c;大家可以通过自己制作H5页面来感受一下…

C语言 | 循环语句总结

C语言循环的嵌套一个循环体内又包含另一个完整的循环结构&#xff0c;称为循环的嵌套。内嵌的循环体中还可以嵌套循环&#xff0c;这就是多层循环。3种常用循环语句&#xff1a;while语句、do...while语句和for语句可以相互嵌套。C语言循环的比较3种循环语句都可以用来处理同一…

python 死循环程序能占满cpu吗_运行tensorflow python程序,限制对GPU和CPU的占用操作...

一般情况下&#xff0c;运行tensorflow时&#xff0c;默认会占用可以看见的所有GPU&#xff0c;那么就会导致其它用户或程序无GPU可用&#xff0c;那么就需要限制程序对GPU的占用。并且&#xff0c;一般我们的程序也用不了所有的GPU资源&#xff0c;只是强行霸占着&#xff0c;…

java 不规则 拼图_Java中不一致的操作会扩大规则

java 不规则 拼图总览 当您在Java中执行一元或二进制运算时&#xff0c;标准行为是使用最宽的操作数&#xff08;或对于byte &#xff0c; short和char &#xff0c;使用更大的操作数&#xff09;。 这很容易理解&#xff0c;但是如果考虑最佳类型可能会造成混淆。 乘法 当执行…

账户的配置使您无法使用该计算机,2个方法解决“user profile service服务未能登录无法加载用户配置文件”...

win8/10系统&#xff1a;启动电脑显示windows标志时&#xff0c;长按电源键强行关机&#xff0c;重复此操作三次&#xff0c;系统将会进入“自动修复”。然后在“自动修复”界面中&#xff0c;选择“高级选项”>“疑难解答”>“高级选项”>“启动设置”>“重启”&a…

endp 汇编start_飞思卡尔Kinetis L 汇编语言启动文件startup_MK25Z4简单分析

打开Freedom KL25的例程包&#xff0c;任意找一个code example&#xff0c;以lower_power_dma_uart_demo为例&#xff0c;分析一下启动汇编文件的代码。代码主要分三部分&#xff1a;堆栈等的配置、向量表、初始启动跳转。1. 堆栈配置; Stack Configuration; Stack Size (in …

自定义ui_如何允许用户自定义UI

自定义ui理念 利用JavafX / FXML的声明性设计模式&#xff0c;并允许用户仅通过使用例如SceneBuilder打开某个视图即可重新定制布局或添加新控件&#xff0c;甚至根据用户需要更改样式&#xff0c;从而无需任何编码即可自定义某个视图。 FXML文件 CSS基本上可以放置在通过URL可…

C语言 | 内部与外部函数

C语言函数的声明和定义一个函数一般由两部分组成&#xff1a;声明部分执行语句C语言对变量而言&#xff0c;声明与定义的关系稍微复杂一些&#xff0c;在声明部分出现的变量有两种情况&#xff1a;一种是需要建立存储空间的。另一种是不需要建立存储空间的。前者称为定义性声明…

进程和线程计算机组成原理面试题,2016年云南财经大学信息学院计算机组成原理复试笔试最后押题五套卷...

一、选择题1&#xff0e; 设文件索引节点中有7个地址项&#xff0c;其中4个地址项为直接地址索引&#xff0c;2个地址项是一级间接地址索引&#xff0c;1个地址项是二级间接地址索引&#xff0c;每个地址项大小为4字节&#xff0c;若磁盘索引块和磁盘数据块的大小均为256字节&a…

获取客户端ip_代理IP工具能否解决反爬?

互联网已成了生活中的部分&#xff0c;从事互联网的工作者&#xff0c;避免不了需要去一些网站上进行爬取需要的数据来达到自己产品或者业务上的需求。比如反爬策略&#xff0c;但是&#xff0c;使用代理IP工具一定可以解决反爬虫策略吗&#xff1f;一、不同的网站有不同的反爬…

C语言 | 结构体数组

C语言结构体数组概述一个结构体变量中可以存放一组有关联的数据&#xff0c;如一个学生的学号、姓名、成绩等数据&#xff0c;如果有10个学生的数据需要参加运算&#xff0c;显然应该用数组&#xff0c;这就是结构体数组。结构体数组与之前介绍的数值型数组的不同之处在于每个数…

html弹窗代码y\/n,Nodejs扩展,实现消息弹窗(示例代码)

模块的C代码 node_gtknotify.cc#include #include #include #include #include using namespace v8;class GtkNotify : node::ObjectWrap{public:GtkNotify(){}~GtkNotify(){}std::string title;std::string icon;static Persistent persistent_function_template;static void …