Java程序优化

Posted 阿拉天啦噜

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Java程序优化相关的知识,希望对你有一定的参考价值。

一. 字符串优化处理

1. String对象组成:char数组,offset偏移量,count长度;

2. String对象特点:

不变性:String对象一旦生成,则不能再对它进行改变;

针对常量池的优化:当两个String对象拥有相同的值时,他们只引用常量池中的同一个拷贝;

类的final定义;

3. subString()

subString()方法之所以会引起内存泄漏,是因为调用了String(int offset, int count, char value[])构造函数,此构造函数采用了用空间换时间的手段,截取字符串时原生内容被复制到了新的字符串中;

4. 字符串的分割与查找

String.split()方法使用简单,功能强大,但是在性能敏感的系统中频繁使用这个方法是不可取的;

StringTokenizer性能优于split()方法;

StringTokenizer st = new StringTokenizer(orgStr, ";");
for(int i = 0; i < 10000; i ++) {
    while (st.hasMoreTokens()) {
        st.nextToken();
    }     
    st = new StringTokenizer(orgStr, ";")    
}

使用indexOf()和subSting()实现的分割算法性能最好,可读性和可维护性最差;

String tmp = orgStr;
for(int i = 0; i < 10000; i ++) {
    while(true) {
        String splitStr = null;
        int j = tem.indexOf(;);
        if(j < 0) break;
        splitStr = tmp.subString(0, j);
        tmp = tmp.subString(j + 1);
    }
    tmp = orgStr;
}

5. charAt()

orgStr.charAt(0) == ‘a‘ && orgStr.charAt(0) == ‘b‘ && orgStr.charAt(0) == ‘c‘ 效率高于orgStr.startsWith("abc");

6. StringBuffer和StringBuilder

对于String常量的累加操作,在编译时会进行优化,将多个连接操作的字符串在编译时合成一个单独的长字符串,并不会生成多个String实例;

对于String变量的累加操作,也做了相应的优化操作,使用了StringBuilder对象来实现字符串的累加;

对于String操作,类似于“+”和“+=”的运算符应该尽量少用,concat()方法的效率远远高于“+”和“+=”运算符,但又低于StringBuilder类;

StringBuilder的效率高于StringBuffer,但无法保证线程安全;

StringBuilder/StringBuffer的默认容量是16字节,超出容量需要扩容,扩容策略是将原有的容量大小翻倍,申请新的内存空间,将原数组中的内容复制到新数组,所以如果能预估StringBuilder/StringBuffer的大小,将能够有效的节省这些操作;

二. 核心数据结构

1. List接口

ArrayList和Vector使用了数组实现,初始值大小为10,每次扩容为1.5倍,唯一区别是Vector是线程安全的;

LinkedList使用了循环双向链表的数据结构;

增加元素到列表尾端:因为数组的连续性,总是在尾端增加元素时,只有在空间不足时才产生数组扩容和数组复制,大部分的追加操作效率非常高,LinkedList每次增加元素都需要新建一个Entry对象,并进行更多的赋值操作,对性能有一定的影响;

增加元素到列表任意位置:插入的元素在List中的位置越靠前,数组重组的开销就越大,可以考虑用LinkedList代替ArrayList;

删除任意位置元素:在ArrayList的每一次删除操作后都需要进行数组的重组,位置越靠前,开销越大;对于LinkedList,双向链表在查找要删除的元素时效率很高,但是移除中间元素则几乎要遍历半个List,在数据量很大时效率很低;所以在删除头或尾部的数据时,可以采用LinkedList,移除中间的元素时使用ArrayList;

对于LinkedList进行随机访问时,总会进行一次列表的遍历操作,所以使用for循环遍历的性能是很差的;forEach在编译后比迭代器多了一步赋值操作,导致性能略差一些;

2. Map接口

HashMap是将key做hash算法,将hash值映射到内存地址,直接取得key对应的数据;底层采用数组实现,内存地址即数组的下标索引;

LinkedHashMap是一个维护了元素次序表的HashMap,可以提供元素插入时的顺序或最近访问的顺序两种实现;

所有的集合都不允许在迭代器模式中修改集合的机构,否则会抛出ConcurrentModificationException,LinkedHashMap在根据元素对吼访问时间进行排序的模式中,没当使用get()方法访问某一元素时,该元素便被移动到链表的尾端,所以此时不能在迭代器中使用get()操作;

TreeMap内部基于红黑树实现,可以根据key进行排序;

3. Set接口

所有Set的实现,都只是对应的Map的一种封装而已;

三. 使用NIO提升性能

1. Channel是一个双向通道,既可读,也可写,但不能直接对Channel进行读写操作,必须通过Buffer来进行;

2. Buffer中有3个重要参数:位置position,容量capacity,上限limit;

3. MappedByteBuffer可以大大提高读取和写入文件的速度,可以适当使用这种方式;

4. 在对普通的ByteBuffer访问时,系统总是会使用一个“内核缓冲区”进行间接的操作,而DirectBuffer所处的位置就相当于这个“内核缓冲区”,由于是更加接近系统底层的方法,所以速度更快;

5. 创建和销毁DirectBuffer的成本较高,所以在频繁创建Buffer的场合不适用,如果可以复用DirectBuffer,可以大幅改善性能;

四. 引用类型

强引用:直接访问目标对象;不会被系统回收;可能导致内存泄漏;

软引用:当堆使用率接近阈值时才会被JVM回收,可以用于实现对内存敏感的Cache;

弱引用:在系统GC时,只有发现弱引用,便会进行回收;软引用,弱引用都可用于保存可有可无的缓存数据,内存不足时可以被回收,内存充足时用来提高访问速度;

虚引用:必须和引用队列一起使用,它的作用在与跟踪垃圾系统回收过程;

WeakHashMap使用弱引用,可以用作缓存,可以自动释放已经被回收的key所在的表项,但如果key都在系统内持有强引用,那么WeakHashMap就退化为普通的HashMap;

五. 有助于改善性能的技巧

1. try-catch语句对系统性能而言是非常糟糕的,尤其放在循环体内;

2. 调用方法时传递的参数以及在调用中创建的临时变量都保存在栈(Stack)中,速度较快,其他变量,如静态变量,实例变量等,都在堆(Heap)中创建,速度较慢;

3. 尝试使用位运算代替部分算数运算,最典型的就是对于整数的乘除运算优化;

4. 使用数组或枚举类代替Switch语句;

5. 一维数组代替二维数组;

6. 提取表达式,减少重复劳动,尤其是循环体中;

7. 展开循环,减少循环次数,提升系统性能;

int[] array = new int[9999999];
for (int i = 0; i < 9999999; i ++) {
     array[i] = i; // 赋值操作
}

for (int i = 0; i < 9999999; i += 3) { // 展开循环,一个循环体完成3个循环的工作
    array[i] = i;
    array[i+1] = i + 1;
    array[i+2] = i + 2;
}

8. 数组复制,System.arrayCopy()函数是native函数,通常native函数的性能要优于普通的函数;

9. 使用new关键字创建轻量级对象时,速度非常快,但是对于重量级对象,由于对象在构造函数中可能会进行一些复杂且耗时的操作,因此构造函数的执行时间可能会比较长,Object.clone()方法可以绕过对象的构造函数,快速复制一个对象实例,默认情况下,生成的实例只是原对象的浅拷贝,如果需要深拷贝,需要重新实现clone()方法;

10. 由于实例方法需要维护一张类似虚函数表的结构,以实现对多态的支持,与静态方法比,实例方法的调用需要更多的资源,对于一些常用的工具类方法,没有对其进行重载的必要,可以声明为static,调用时也不需要生成类的实例,比调用实例方法更为方便,易用;

以上是关于Java程序优化的主要内容,如果未能解决你的问题,请参考以下文章

如何在片段中使用 GetJsonFromUrlTask​​.java

java代码在片段活动中不起作用

使用 C++ 反转句子中的每个单词需要对我的代码片段进行代码优化

如何优化C ++代码的以下片段 - 卷中的零交叉

LockSupport.java 中的 FIFO 互斥代码片段

Android 逆向整体加固脱壳 ( DEX 优化流程分析 | DexPrepare.cpp 中 dvmOptimizeDexFile() 方法分析 | /bin/dexopt 源码分析 )(代码片段