5个优化java代码的性能的tips

大多数时候,应用性能的优化不是必需的,但是本文包含的5种办法非常简单,可以在代码开发期间低成本得采纳,以防止java程序变慢以及占用更多的资源。

尽可能于设置HashMap和ArrayList的大小

HashMap和ArrayList等基于数组的结构,都存在一个问题,就是当不断添加数据超过阈值时,就会触发resize扩容操作,尽管这些类的大多数操作都很快,比如ArrayList.get(index)是 o(1)时间复杂度,但是resize操作的时间都是O(n),并且可能会出现很多次的重复。

ArrayList 默认的容量为10,当数据超过10时候,就会触发扩容,扩容操作为new_capacity = old_capacity * 2;此时会变成20大小的数组。

假如我们需要添加500个元素到ArrayList中,那么它的内部表现是这样的:10->20->40->80->160->320->640. 总共产生了6次扩容操作,每次操作都会复制一次整个数组,同时最后产生了一个640大小的数组,而其实我们只需要500大小的数组,这里就导致了21%空间的浪费。

如果我们能明确值存储500个对象,就可以将ArrayList大小直接设定为500。

ini 复制代码
List<Object> list = new ArrayList(500);

这样就可以避免使用过程中的扩容操作以及浪费的45%内存(20大小的数组,只使用了11个位置,浪费了9个)。

HashMap的扩容情况比ArrayList更严重,因为它含有一个扩容因子参数,默认为0.75,当数据量达到容量的0.75时就会触发扩容。

因为有扩容因子的存在,对于HashMap大小的设置会有一点麻烦,推荐使用Guava的Maps.newHashMapWithExpectedSize(int size)方法,该方法内置了预期大小的计算.

ini 复制代码
Map<String> map = Maps.newHashMapWithExpectedSize(24);

对HashMap的复合key使用实体类包装

当我们在HashMap中使用复合的String作为key时,使用一个实体类可以加快速度和避免内存分配。

错误方式

ini 复制代码
    String[] prefixes;
    String[] suffixes;
    Map<String, Object> concatMap;
    // 获取数据
      for (int i = 0; i < prefixes.length; ++i) {
        concatMap.get(prefixes[i] + ";" + suffixes[i]);
      }

正确方式

ini 复制代码
    String[] prefixes;
    String[] suffixes;
    Map<Pair, Object> pairMap;
    // 获取数据
      for (int i = 0; i < prefixes.length; ++i) {
        pairMap.get(new Pair(prefixes[i], suffixes[i]));
      }

正确方式的性能是错误方式的3.7倍。

使用ThreadLocalRandom 替换 Random

Random 是java开发者非常熟悉的类,它的作用是用来生成随机数。它在功能上没有什么问题,唯一的问题是因为那部使用同步代码块,多个线程并发时,就会出现竞争、冲突问题,导致效率变慢。

ThreadLocalRandom 是JDK1.7中新推出的基于ThreadLocal的Radnom类,实现了每个线程拥有自己的Random类,避免了多线程时的冲突。

在JDK1.7版本以后,在任何场景都应该优先使用ThreadLocalRandom,它只有好处没有坏处。

使用debug日志时,避免不必要的函数调用

java中最有流行的日志框架当属Slf4j,其中就提供了debug方法来记录日志。debug 方法提供了占位符来避免在非debug日志等级时的字符串拼接。

logger.debug("Printing variable value: {}", variable);

但是在某些使用错误的场景中,就会带来意想不到的资源浪费。比如调用了json.toString();

ini 复制代码
JSONObject json = xxx;
logger.debug("Printing variable value: {}", json.toString());

这种使用方法会导致json.toString()会被调用,但是最后又不输出。使用logger.isDebugEnabled可以消除该问题。

ini 复制代码
JSONObject json = xxx;
if (logger.isDebugEnabled()) {
    logger.debug("Printing variable value: {}", json.toString());
}

停止使用JDK8

JDK8后续的LTS版本11中,包含了非常多的优化,比如String类的优化,GC的优化等。所有的java应用都大量使用String,在JDK8中,String比较大而且慢。在JDK8中,String使用char[]进行存储。

arduino 复制代码
private final char[] value;

JDK9以后变成了byte[];

arduino 复制代码
private final byte[] value;

一系列存储和编码的优化减少了String存储的大小和编码解码的速度。

总结

本文介绍了5种简单有效的java性能优化手段,以方便大家在工作写出更高效的程序

相关推荐
星就前端叭6 分钟前
【开源】一款基于SpringBoot的智慧小区物业管理系统
java·前端·spring boot·后端·开源
带刺的坐椅7 分钟前
RxSqlUtils(base R2dbc)
java·reactor·solon·r2dbc
silence25020 分钟前
深入了解 Reactor:响应式编程的利器
java·spring
weixin_SAG30 分钟前
21天掌握javaweb-->第19天:Spring Boot后端优化与部署
java·spring boot·后端
m0_7482475533 分钟前
SpringMVC跨域问题解决方案
java
Elcker34 分钟前
KOI技术-事件驱动编程(Sping后端)
java·spring·架构
GitNohup37 分钟前
Spring boot处理跨域问题
java·spring boot·跨域
Just_Paranoid1 小时前
使用 IDE生成 Java Doc
java·开发语言·ide
西海天际蔚蓝1 小时前
递归查询全量分页数据问题
java
俎树振1 小时前
深入理解与优化Java二维数组:从定义到性能提升的全面指南
java·算法