Java代码优化建议

今天看了一篇微信推送,关于优化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();”会比较合适一些


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值