今天看了一篇微信推送,关于优化Java代码的建议,总结如下:
1、尽量指定类、方法的final修饰符
带有final修饰符的类是不可派生的。在Java核心API中,有许多应用final的例子,例如java.lang.String,整个类都是final的。为类指定final修饰符可以让类不可以被继承,为方法指定final修饰符可以让方法不可以被重写。如果指定了一个类为final,则该类所有的方法都是final的。Java编译器会寻找机会内联所有的final方法,内联对于提升Java运行效率作用重大,具体参见Java运行期优化。此举能够使性能平均提高50%。
2、尽量重用对象
特别是String对象的使用,出现字符串连接时应该使用StringBuilder/StringBuffer代替。由于Java虚拟机不仅要花时间生成对象,以后可能还需要花时间对这些对象进行垃圾回收和处理,因此,生成过多的对象将会给程序的性能带来很大的影响。
3、尽可能使用局部变量
调用方法时传递的参数以及在调用中创建的临时变量都保存在栈中,速度较快,其他变量,如静态变量、实例变量等,都在堆中创建,速度较慢。另外,栈中创建的变量,随着方法的运行结束,这些内容就没了,不需要额外的垃圾回收。
4、及时关闭流
5、尽量减少对变量的重复计算
明确一个概念,对方法的调用,即使方法中只有一句语句,也是有消耗的,包括创建栈帧、调用方法时保护现场、调用方法完毕时恢复现场等。所以例如下面的操作:
1
2
|
for
(
int
i =
0
; i < list.size(); i++)
{...}
|
建议替换为:
1
2
|
for
(
int
i =
0
, length = list.size(); i < length; i++)
{...}
|
这样,在list.size()很大的时候,就减少了很多的消耗
6、如果能够估计到待添加的内容长度,为底层以数字方式实现的集合工具类指定初始长度
集合工具类,如StringBuilder、StringBuffer等初始化长度为16,如果超过该大小,会以当前2倍再加2的速度扩展数组大小,并且扩展后会将原来的数据复制到新扩展的数组中,这样十分消耗性能。
7、循环内不要不断创建对象引用
可以在循环外创建应用,例如
1
2
3
4
|
for
(
int
i =
1
; i <= count; i++)
{
Object obj =
new
Object();
}
|
这种做法会导致内存中有count份Object对象引用存在,count很大的话,就耗费内存了,建议为改为:
1
2
3
4
5
|
Object obj =
null
;
for
(
int
i =
0
; i <= count; i++)
{
obj =
new
Object();
}
|
这样的话,内存中只有一份Object对象引用,每次new Object()的时候,Object对象引用指向不同的Object罢了,但是内存中只有一份,这样就大大节省了内存空间了。
8、基于效率和类型的检查考虑,应尽可能使用array,无法确定数组大小时才使用ArrayList
9、尽量使用HashMap,ArrayList、StringBuilder,除非线程安全需求,否则不推荐使用HashTable、Vector、StringBuffer,后三者由于使用同步机制而导致了性能开销
10、尽量避免随意使用静态变量
要知道,当某个对象被定义为static的变量引用,那么gc通常不会回收这个对象所占有的堆内存的:
public
class
A
{
private
static
B b =
new
B();
}
此时静态变量b的生命周期和A类相同,如果A类不被卸载,那么引用B指向的B对象会常驻内存,直到程序终止
11、顺序插入和随机访问比较多的场景使用ArrayList,元素删除和中间插入比较多的场景使用LinkedList
12、字符串变量和字符串常量equals的时候,将字符串常量写在前面,这样做可以避免空指针异常
13、不要对数组使用toString()方法
14、使用最有效的方法去遍历Map
遍历Map的方式有很多,通常场景下我们需要的是遍历Map中的Key和Value,那么推荐使用的、效率最高的方式是:
1
2
3
4
5
6
7
8
9
10
11
12
13
|
public
static
void
main(String[] args)
{
HashMap<String, String> hm =
new
HashMap<String, String>();
hm.put(
"111"
,
"222"
);
Set<Map.Entry<String, String>> entrySet = hm.entrySet();
Iterator<Map.Entry<String, String>> iter = entrySet.iterator();
while
(iter.hasNext())
{
Map.Entry<String, String> entry = iter.next();
System.out.println(entry.getKey() +
""
+ entry.getValue());
}
}
|
如果你只是想遍历一下这个Map的key值,那用”Set<String> keySet = hm.keySet();”会比较合适一些