11 个简练的 Java 性能调优技巧

转载自 11 个简练的 Java 性能调优技巧

想要让你的项目一直高性能运作吗?以下有一些技巧你可以拿去消除缓存瓶颈,还有一些其他的性能调优建议。

大多数开发者认为性能优化是一个复杂的话题,它需要大量的工作经验和相关知识理论。好吧,这也不完全错。优化一个应用做到性能最优化可能不是件容易的任务,但是这并不意味着你没有相关的知识就什么也做不了。这里有一些易于遵循的建议和最佳实践可以帮助你创建一个性能良好的应用程序。

这些建议的大部分都是针对 Java 语言的。但是也有一些是跟语言无关的,你可以运用到任意的应用和程序中。在我们学习特定的 Java 编程性能调优之前,先来探讨一些通用的技巧。

1. 在明确必要之前别急着优化

这可能是最重要的性能优化技巧之一。你应该遵循常见的最佳实践做法并在案例中高效地应用它。但是这并不意味在证明必要之前,你应该更换任何标准库或构建复杂的优化。

多数情况下,过早地优化会占用大量的时间,而且会使代码变得难以理解和阅读。更糟糕的是,这些优化通常并没带来任何好处,因为你花了大量的时间在优化应用中的非关键部分。

那么,要怎么证明东西需要优化呢?

首先,你需要定义你的代码速度得多快。例如,为所有 API 调用指定最大响应时间,或者指定在特定时间范围内要导入的记录数量。在做完这些后,你需要确定你应用中哪些部分太慢需要改进。当完成这些后,你就可以来看看第二个技巧提示。

2. 使用分析器找到真正的瓶颈

在完成第一部分的优化建议以鉴别出你应用中需要提升的部分后,要从哪里入手呢?

你可以有两种途径来解决这个问题:

1)查看你的代码,从看起来可疑的或者你觉得可能会导致出现问题的地方入手

2)或者使用分析器获取代码每个部分的行为(执行过程)和性能的详细信息。

希望我不需要解释为什么应该始终遵循第二种途径/方法的原因。

很显然,基于分析器的方式可以让你更好地理解代码的性能影响,并允许你去专注于更关键的部分(代码)。即使你曾经使用过分析器,你一定记得你曾经多么惊讶于一下就找到了代码的哪些部分产生了性能问题。我第一次的猜测不止一次地导致我走错了方向。

3. 为整个应用程序创建一个性能测试套件

这是另一个通用的可以帮助你避免在将性能改进部署到产品中之后经常会发生的许多意外问题的技巧。你应该总是定义一个性能测试套件来测试整个应用程序,并在性能改进之前和之后运行它。

这些额外的测试运行将帮助你识别你的改动所引起的功能和性能上的副作用,并确保不会导致弊大于利的更新。如果你处理的是被应用程序的多个不同部分使用的组件,如数据库或缓存,那这一点尤为重要。

4. 优先关注最大瓶颈

在创建了测试套件并使用分析器分析你的应用程序之后,你可以列出一系列需要解决以提高性能的问题列表。这很好,但这并没有回答你需要从哪里开始的问题。你可以专注于速成方案,或从最重要的问题开始。

速成方案一开始可能会很有吸引力,因为你可以很快显示第一个成果。但有时,可能有必要说服其他团队成员或管理层认为性能分析是值得的。

一般来说,我建议从顶层开始,首先开始处理最重要的性能问题。这将为你提供最大的性能改进,而且你可能仅需要解决这些问题中的一小部分就能满足你的性能要求。

常见的通用调优技巧到此结束。接下来让我们仔细看看一些特定于 Java 的技巧。

5. 使用 StringBuilder 以编程方式连接字符串

在 Java 中有很多不同的选项来连接字符串。例如,你可以使用简单的 + 或 + = ,以及老的 StringBuffer 或 StringBuilder 。

那么,你应该选择哪种方法呢?

答案取决于连接字符串的代码。如果你是以编程方式将新内容添加到字符串中,例如在 for 循环中,则应使用 StringBuilder 。它很易于使用,并提供比 StringBuffer 更好的性能。但请记住,与 StringBuffer 相比, StringBuilder 不是线程安全的,可能并不适用于所有情况。

你只需要实例化一个新的 StringBuilder 并调用 append 方法来向 String 中添加一个新的部分。在你添加完了所有的部分后,你可以调用 toString() 方法来检索已连接的字符串。

下面的代码片段展示了一个简单的例子。在每次迭代期间,该循环将 i 转换为一个 String ,并将其与空格一起添加到 StringBuilder sb 中。所以,最后,这段代码在日志文件中写入 “This is a test0 1 2 3 4 5 6 7 8 9” 。

StringBuilder sb = new StringBuilder(“This is a test”);for (int i=0; i<10; i++) {
   sb.append(i);
   sb.append(” “);
}
log.info(sb.toString());


正如你在我们的公众号Java技术栈代码片段中看到的,我们可以为字符串的第一个元素提供到构造函数中。这会创建一个 StringBuilder ,其中包含了你所提供的字符串以及 16 个额外字符的容量。当你向 StringBuilder 中添加更多字符时,你的 JVM 将动态的增加 StringBuilder 的大小。

如果你已经知道字符串将包含多少个字符,则可以将该数字提供给不同的构造方法以实例化具有指定容量的 StringBuilder 。这进一步提高了效率,因为它不需要动态扩展其容量。

6. 使用 + 连接一个语句中的字符串

当你使用 Java 实现你的第一个应用程序时,可能有人告诉过你不要使用 + 来连接字符串。如果你是在应用程序逻辑内连接字符串的话,这是对的。字符串是不可变的,每个字符串的连接结果都被存储在一个新的字符串对象中。这需要额外的存储空间,并可能使你的应用程序运行缓慢,特别是当你在一个循环内连接多个字符串的情况下。

在这些情况下,你应该遵循技巧 5 中的内容,并使用 StringBuilder 。

但如果你只是将字符串分成多行来改善代码的可读性,这并不适用。

Query q = em.createQuery(“SELECT a.id, a.firstName, a.lastName ”
+ “FROM Author a ”
+ “WHERE a.id = :id”);


在这些情景下,你应该使用简单的 + 来连接字符串。你的 Java 编译器会优化它,并在编译时完成连接。因此,在运行时,你的代码将只使用一个字符串,并不需要任何连接操作。

7. 尽可能使用基本类型

避免任何开销并提高应用程序性能的另一种简便快速的方法是使用基本类型而不是其包装类。所以,最好使用 int 而不是 Integer ,是 double 而不是 Double 。这将使得你的 JVM 将值存储在堆栈而不是堆中,以减少内存消耗,并更有效地处理它。

8. 尽量避免大整数和小数

由于我们已经在讨论数据类型,所以我们也应该快速浏览大整数和小数。尤其是后者因其精确性而受欢迎。但这是有代价的。

大整数和小数比一个简单的 long 型或 double 型需要更多的内存,并会显著减慢所有的运算。所以,如果你需要额外的精度,或者如果你的数字超出一个较长的范围,最好要三思。这可能是你需要更改并解决性能问题的唯一方法,尤其是在实现数学算法时。

9. 优先检查当前日志级别

这个建议应该是显而易见的,但不幸的是,很多人在写代码的时候都会忽略它。 在创建调试消息之前,应该总是优先检查当前日志级别。 否则,你可能会创建一个附加你日志消息的字符串,而该字符串之后将被忽略。

这里有两个你不应该这样做的反面例子。

// don’t do this
log.debug(“User [” + userName + “] called method X with [” + i + “]”);
// or this
log.debug(String.format(“User [%s] called method X with [%d]”, userName, i));


在这两个示例中,你都将执行创建日志消息所有必需的步骤,而不知道日志框架是否将使用日志消息。 因此在创建调试消息之前,最好先检查当前的日志级别。

// do thisif (log.isDebugEnabled()) {
   log.debug(“User [” + userName + “] called method X with [” + i + “]”);
}


10. 使用 Apache Commons StringUtils.Replace 而不是 String.replace

一般来说,String.replace 方法可以正常工作,并且效率很高,尤其是在你使用 Java 9 的情况下。但是,如果你的应用程序需要大量的替换操作,并且没有更新到最新的 Java 版本,那么检查更快和更有效的替代品依然是有必要的。

有一种候选方案是 Apache Commons Lang 的 StringUtils.replace 方法。正如 Lukas Eder 在他最近的一篇博客文章中所描述的,它远远胜过了 Java 8 的 String.replace 方法。

而且它只需要很小的改动。你只需要将 Apache Commons Lang 项目的 Maven 依赖项添加到你的应用程序的 pom.xml 中,并将 String.replacemethod 的所有调用替换为 StringUtils.replace 方法。

// replace this
test.replace(“test”, “simple test”);
// with this
StringUtils.replace(test, “test”, “simple test”);


11.昂贵的缓存资源,如数据库连接

缓存是避免重复执行昂贵或常用代码片段的流行解决方案。总的思路很简单:重复使用这些资源比创建一个新的资源更划算。

一个典型的例子是缓存池中的数据库连接。新连接的创建需要时间,如果你重用现有连接,则可以避免这种情况。

你也可以在 Java 语言源码中找到其他的例子。例如,在 Integer 类中的 valueOf  方法缓存了介于 -128 到 127 之间的值。你可能会说创建一个新的 Integer 并不是太昂贵,但是由于它经常被使用,因此缓存最常用的值也可以提供性能优势。

但是,当你考虑使用缓存时,请记住缓存实现也会产生开销。你需要花费额外的内存来储存可重复使用的资源,因此你可能需要管理你的缓存以使资源可访问,并删除过期的资源。

所以,在开始缓存任何资源之前,请确保它们是经常使用的,以超过缓存实现的开销(代价)。

总结

正如你所看到的,有时不需要太多的工作就可以提高你的应用程序的性能。本文中的大部分建议只需要稍作努力就可以将它们应用于你的代码中。

但还是那句话,最重要的还是那些与是什么编程语言无关的技巧:

  • 在你知道其必要性之前不要进行优化

  • 使用分析器(profiler)来查找真正的瓶颈

  • 优先处理最大的瓶颈


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

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

相关文章

pre1-flink理论-批处理与流处理+简单示例

【README】 1.本文包含了 批处理与流处理的代码示例&#xff1b; 批处理&#xff1a;把数据 攒在一起&#xff08;或攒一段时间或攒一定内存大小&#xff09;&#xff0c;然后再处理&#xff0c;这叫批处理&#xff1b;流处理&#xff1a;数据每来一个就处理一个&#xff1b;…

python表单提交的两种方式_Flask框架学习笔记之表单基础介绍与表单提交方式

本文实例讲述了Flask框架学习笔记之表单基础介绍与表单提交方式。分享给大家供大家参考&#xff0c;具体如下&#xff1a;表单介绍表单是HTML页面中负责数据采集功能的部件。由表单标签&#xff0c;表单域和表单按钮组成。通过表单&#xff0c;将用户输入的数据提交给服务器&am…

高级 | Java中获取类名的3种方法

转载自 高级 | Java中获取类名的3种方法获取类名的方法 Java 中获取类名的方式主要有以下三种。 getName() 返回的是虚拟机里面的class的类名表现形式。 getCanonicalName() 返回的是更容易理解的类名表示。 getSimpleName() 返回的是类的简称。 都有什么区别&#xff1f; 通过…

Asp.net 面向接口可扩展框架之核心容器

新框架的容器部分终于调通了&#xff01;容器实在太重要了,所以有用了一个名词叫“核心容器”。 容器为什么那么重要呢&#xff1f;这个有必要好好说道说道。 1、首先我们从框架名称面向接口编程说起,什么是面向接口编程&#xff1f;(这个度娘回答一下) 解读一下:类是个体的定义…

pre2-flink单机部署与job提交

【README】 本文记录了flink单机部署&#xff0c;以及flink job2种提交方式&#xff1b; 【1】flink 单机部署 step1&#xff09;下载flink 包&#xff1b; Apache Flink: Stateful Computations over Data Streamshttps://flink.apache.org/ step2&#xff09;解压 tar -z…

到底什么是跨域?附解决方案

转载自 到底什么是跨域&#xff1f;附解决方案什么是跨域 要了解跨域&#xff0c;先要说说同源策略。 同源策略是由 Netscape 公司提出的一个著名的安全策略&#xff0c;所有支持 JavaScript 的浏览器都会使用这个策略。 所谓同源是指&#xff0c;域名&#xff0c;协议&#xf…

vue 字典_【开源】基于Vue的前端组件库HeyUI

说道vue组件库&#xff0c;目前主流的基本就是iview和element。今天又发现一个很不错的。HeyUI。组件也很丰富&#xff0c;入门比较简单。反正开源框架我们有不嫌多&#xff0c;多多益善啊。感兴趣的可以看看。关于HeyUIHeyUI 是一套基于 Vue2.0 的开源 UI 组件库&#xff0c;主…

(译)java8-流定义

【README】 本文翻译自 Stream In Java - GeeksforGeeks &#xff0c; 主要介绍了java8流&#xff1b; 【1】流 1&#xff09;流定义&#xff1a;流是支持各种方法的对象序列&#xff08;一系列对象&#xff09;&#xff0c;这些方法可以流水线化调用以产生期望结果&#xff…

基于CefSharp构建基于Chromium的应用程序

chromium是google chrome浏览器所采用的内核&#xff0c;最开始由苹果的webkit发展而出&#xff0c;由于webkit在发展上存在分歧&#xff0c;而google希望在开发上有更大的自由 度&#xff0c;2013年google决定自己开发webcore的分支&#xff0c;叫做Blink引擎&#xff0c;而后…

最新后端架构师技术图谱

转载自 最新后端架构师技术图谱深呼吸&#xff0c;慢慢学&#xff0c;技术长路漫漫… 数据结构二叉树完全二叉树平衡二叉树二叉查找树&#xff08;BST&#xff09;红黑树B-&#xff0c;B&#xff0c;B*树LSM 树队列集合链表、数组字典、关联数组栈树BitSet常用算法KPM 算法选择…

ansible脚本-Playbook(一)

Playbook组成部分&#xff1a; task 任务&#xff1a;包含目标主机上执行的操作&#xff0c;使用模块定义这些操作&#xff0c;每个任务都是一个模块的调用Variables变量&#xff1a;存储和传递数据&#xff0c;变量可以自定义&#xff0c;可以在playbook当中定义为全局变量&a…

三级pc技术_第十九周PC、笔电、数码周边新品汇总:AMD英特尔激战正酣

【dogkeji-科技犬】各位网友周末好&#xff0c;又到了2020年第十九周的PC、笔电、数码周边新品发布汇总时刻&#xff08;2020年5月4日至2020年5月9日&#xff09;&#xff0c;那么本周有那些PC、笔电、数码周边新品发布呢&#xff1f;通过科技犬的汇总我们来一起回顾一下吧。AM…

【DDD/CQRS/微服务架构案例】在Ubuntu 14.04.4 LTS中运行WeText项目的服务端

在《WeText项目&#xff1a;一个基于.NET实现的DDD、CQRS与微服务架构的演示案例》文章中&#xff0c;我介绍了自己用Visual Studio 2015&#xff08;C# 6.0 with .NET Framework 4.6.1&#xff09;开发的DDD/CQRS/微服务架构的案例项目&#xff1a;WeText。文章发出后反响很好…

es6 dsl与sql对比

【README】 1.本文总结了 dsl 与 sql的对比写法&#xff1b; 2.es采用 7.2.1 版本&#xff1b; 【1】创建es索引 1&#xff09;新建一个数据库事务执行日志索引 put localhost:9200/txlog { "mappings" :{ "properties":{"APPNAME"…

echarts line 去掉最外围方框_干货 | 关于射频芯片最详细解读

传统来说&#xff0c;一部可支持打电话、发短信、网络服务、APP应用的手机&#xff0c;一般包含五个部分部分&#xff1a;射频部分、基带部分、电源管理、外设、软件。射频部分&#xff1a;一般是信息发送和接收的部分&#xff1b;基带部分&#xff1a;一般是信息处理的部分&am…

服务器性能指标(一)——负载(Load)分析及问题排查

转载自 服务器性能指标&#xff08;一&#xff09;——负载&#xff08;Load&#xff09;分析及问题排查平常的工作中&#xff0c;在衡量服务器的性能时&#xff0c;经常会涉及到几个指标&#xff0c;load、cpu、mem、qps、rt等。每个指标都有其独特的意义&#xff0c;很多时候…

HoloLens开发手记 - HoloLens shell概述 HoloLens shell overview

使用HoloLens时&#xff0c;shell是由你周围的世界和来自系统的全息图像构成。我们将这种空间成为混合世界&#xff08;mixed world&#xff09;。 shell包含了一个可以让你将全息图像和应用放置在世界中的开始菜单&#xff08;Start Menu&#xff09;。当一个应用已经被放置在…

【1】flink-source读取数据

【README】 本文记录了flink读取不同数据源的编码方式&#xff0c;数据源包括&#xff1b; 集合&#xff08;元素列表&#xff09;&#xff1b;文件kafka&#xff1b;自定义数据源&#xff1b; 本文使用的flink为 1.14.4 版本&#xff1b;maven依赖如下&#xff1a; <dep…

Oracle入门(二)之服务启动bat

转载自 批处理&#xff08;bat文件&#xff09;自动启动/关闭oracle服务 批处理&#xff08;bat文件&#xff09; 自动启动/关闭oracle服务 判断oracle 服务状态如果服务处于启动状态&#xff0c;就关闭服务&#xff1b;如果服务处于关闭状态&#xff0c;就启动服务。 ECHO OFF…

【2】flink数据流转换算子

【README】 本文记录了flink对数据的转换操作&#xff0c;包括 基本转换&#xff0c;map&#xff0c;flatMap&#xff0c;filter&#xff1b;滚动聚合&#xff08;min minBy max maxBy sum&#xff09;&#xff1b;规约聚合-reduce&#xff1b;分流&#xff1b;connect连接流…