Java性能优化的十条小技巧
1 System.nanoTime
测试性能时,System.nanoTime比System.currentTimeMills更精确,前者使用纳秒计时,且对系统影响更小。
具体来说:
System.currentTimeMills返回自1970年1月1日以来经过的毫秒数,返回的精度与操作系统有关System.nanoTime:不是现实时间,是虚拟机提供的计时时间,精确到纳秒
2 ThreadLocalRandom
通常生成随机数会使用Random类,Random是线程安全的,Random实例里面有一个原子性的种子变量来记录当前种子的值,当要生成新的随机数时,会根据当前种子计算新的种子并更新回原子变量。多线程下计算新种子,会竞争同一个原子变量的更新操作,会造成大量线程进行自旋测试,降低并发性能。
而ThreadLocalRandom在当前线程维护了一个种子,适合在多线程场景下提供高性能的伪随机数生成,使用如下:
ThreadLocalRandom random = ThreadLocalRandom.current();
random.nextInt(range);
3 使用局部变量
理论上说,访问局部变量会快于类变量,因为局部变量保存在方法栈中,而类变量保存在堆中。如果在某个类方法中需要多次访问类变量,建议先创建一个局部变量并使其具有与类变量相同的值。
4 关于正则表达式替换
由于正则表达式替换时每次都需要编译正则表达式到一个中间结构,因此比常规的直接替换要慢,如果是固定的正则表达式替换,可以采用预编译的思想:
Pattern pattern = Pattern.compile("origin str");
public String replace(String str){
return pattern.matcher(str).replaceAll("target str");
}
而不是采用:
public String replace(String str){
return str.replace("origin str","target str");
}
5 关于字符串拼接
尽可能使用如下形式:
String a = "xxx";
String b = "xxx";
String c = new StringBuilder().append(a).append(b).toString();
性能相对不好的是如下情形(得益于JVM默认开启字符串拼接优化):
String c = a+b;
性能最差的是:
StringBuilder c = new StringBuilder();
c.append(a);
c.append(b);
String result = c.toString();
因为这样JIT不会优化。
另外,在无关线程安全的情况下,尽可能使用StringBuilder而不是StringBuffer。
6 关于数字转字符串
int转String是一个较为耗时的操作,尽量避免不必要的转化,如果确实需要,可以预先将一批int转为String,需要的时候直接取出:
public static class CommonUtil{
static int cacheSize = 1024;
static String [] caches = new String[cacheSize];
static{
for(int i=0;i<cacheSize;++i){
caches[i] = String.valueOf(i);
}
}
public static String int2String(int data){
if(data < cacheSize){
return caches[size];
}else{
return String.valueOf(data);
}
}
}
这样相比起直接使用
Stirng.valueOf(data)
性能会高一点。
7 switch/if
少分支的情况下,建议使用if,多分支建议使用switch,常用的“少分支”标准是2-5个。
8 采用返回码而不是抛异常
除非必要使用异常,应该避免把正常的返回错误结果使用异常来代替。抛异常会导致性能是因为构造异常对象时需要一个填写异常栈的过程,就是Throwable中的fillInStackTrace,这是一个Native方法,会填写异常栈,造成较为严重的耗时。
一种优化方法是,自定义异常,重写fillInStackTrace():
public class MyException extends RuntimeException{
...
public synchronized Throwable fillInStackTrace(){
this.setStackTrace(new StackTraceElement[0]);
return this;
}
}
另外,JVM会对频繁抛出的异常做Fast Throw优化,如果检测到代码中某一位置连续多次抛出同一类型的异常,则采用Fast Throw方式,异常栈信息不会被填写,这种异常抛出速度很快,因为不需要在堆里分配内存,也不需要构造完整的异常栈信息,默认对如下异常采用Fast Throw优化:
NullPointerExceptionArithmeticExceptionArrayIndexOutOfBoundsExpcetionArrayStoreExceptionClassCastException
需要注意的是,Fast Throw虽然提高了性能,但是会导致异常栈消息,从而无法快速定位到错误代码,如果需要避免异常栈优化,可以使用参数:
-XX:-OmitStackTraceInFastThrow
9 位运算
可以通过位运算代替部分算术运算以提高性能,比如:
- 判断奇数:
(a & 1) == 1 - 判断偶数:
(a & 1) == 0 - 除2:
a>>1 - 乘2:
a<<1
10 其他技巧
- 字符串搜索等需要搜索单个字符时,使用
String.indexOf(char)而不是String.indexOf(String) - 对于判断一些特殊的
ID,比如长度9位且以11开头,可以直接使用常数判断:id>=110_000_000 && id<=120_000_000,而不需要通过String.valueOf转为字符串再通过String.length+String.startWith判断 - 在
switch中,可以使用int去代替String - 日志输出可以直接使用字符串拼接而不是
模板+{},因为会有一个占位符{}替换成目标变量的耗时过程,被频繁调用的话建议直接字符串拼接 - 传输的实体类尽量避免使用
String,因为其中涉及序列化、反序列化、字符串构造,而对于byte[]构造String的方法,内部会调用StringCoding.decode,相比起通过char[]/Stirng构造会造成更大的耗时
Java性能优化的十条小技巧的更多相关文章
- UITableView的性能优化10个小技巧
通常你会发现一个图片类的app会在一个imageView上做下面这些事情: 1 下载图片(主要的内容图片+用户头像图片)2 更新时间戳3 展示评论4 计算动态的cell的高度 Tip#1 学习 ...
- iOS-UITableView的性能优化10个小技巧
通常你会发现一个图片类的app会在一个imageView上做下面这些事情: 1 下载图片(主要的内容图片+用户头像图片)2 更新时间戳3 展示评论4 计算动态的cell的高度 Tip#1 学习 ...
- Java 性能优化手册 — 提高 Java 代码性能的各种技巧
转载: Java 性能优化手册 - 提高 Java 代码性能的各种技巧 Java 6,7,8 中的 String.intern - 字符串池 这篇文章将要讨论 Java 6 中是如何实现 String ...
- Java 性能优化的五大技巧
要对你的 Java 代码进行优化,需要理解 Java 不同要素之间的相互作用,以及它是如何与其运行时的操作系统进行交互的.使用下面这五个技巧和资源,开始学习如何分析和优化你的代码吧. 在我们开始之前, ...
- 《Java性能优化权威指南》
<Java性能优化权威指南> 基本信息 原书名:Java performance 原出版社: Addison-Wesley Professional 作者: (美)Charlie Hunt ...
- 推荐:Java性能优化系列集锦
Java性能问题一直困扰着广大程序员,由于平台复杂性,要定位问题,找出其根源确实很难.随着10多年Java平台的改进以及新出现的多核多处理器,Java软件的性能和扩展性已经今非昔比了.现代JVM持续演 ...
- 【转】10种简单的Java性能优化
10种简单的Java性能优化 2015/06/23 | 分类: 基础技术 | 14 条评论 | 标签: 性能优化 分享到: 本文由 ImportNew - 一直在路上 翻译自 jaxenter.欢迎加 ...
- 44个Java性能优化
44个Java性能优化 首先,代码优化的目标是: 减小代码的体积 提高代码运行效率 代码优化细节 1 .尽量指定类.方法的final修饰符 带有final修饰符的类是不可派生的.在Java核心AP ...
- java 性能优化(代码优化)
参考博文: java 性能优化:35 个小细节,让你提升 java 代码的运行效率
随机推荐
- Warning: Cannot update during an existing state transition (such as within `render`). Render 报错
原来 修改(不用在构造函数里面定义)
- 深入剖析 ConcurrentHashMap
自建博客地址:https://bytelife.net,欢迎访问! 本文为博客自动同步文章,为了更好的阅读体验,建议您移步至我的博客 本文作者: Jeffrey 本文链接: https://bytel ...
- idea将文件push之后如何回退
- IntelliJ Idea Error Address localhost 1099 is already in use.
Reference: https://stackoverflow.com/questions/38986910/intellij-idea-address-localhost1099-is-a ...
- SpringBoot(八):SpringBoot中配置字符编码 Springboot中文乱码处理
SpringBoot中配置字符编码一共有两种方式 方式一: 使用传统的Spring提供的字符编码过滤器(和第二种比较,此方式复杂,由于时间原因这里先不介绍了,后续补上) 方式二(推荐使用) 在appl ...
- redis数据结构和对象二
跳跃表(skiplist) 跳跃表是一种有序数据结构.跳跃表支持平均O(logN),最坏O(N)复杂度的节点查找,大部分情况下,跳跃表的效率可以和平衡树相媲美,并且因为跳跃表的实现比平衡树简单,所有不 ...
- WPF 基础 - 控件与布局
1. 前言 1. 数据驱动 UI WPF 是数据核心.主动的,UI 从属数据并表达数据.是被动的: 不再是 UI 驱动数据,而是数据驱动 UI: 2. 控件的定义 控件.数据内容.行为(控件响应用户操 ...
- python基础学习之类
面向对象和面向过程 面向过程:以吃饭为例,即为 煮饭.洗菜.洗碗.切菜.炒菜.出锅.吃饭面向对象:目标对象做完,直接吃疑问点:1.面向对象就是把过程用函数封装起来,随时调用?:2.面向过程就是每次都把 ...
- webpack4.x 从零开始配置vue 项目(一)基础搭建项目
序 现在依旧记得第一次看到webpack3.x 版本配置时候的状态 刚开始看到这些真的是一脸懵.希望这篇文章能帮到刚开始入门的同学. webpack 是什么? webpack是一个模块化打包工具,w ...
- java IO流文件拷贝文件(字符流标准写法)
public static void copyFile2(String path1, String path2) { Reader reader = null; Writer writer = nul ...