代码的优化 性能的提升 - JAVA

这篇将作为我这一段时间以来结合项目、学习对于自我提升的过程中的一个技术性的经验总结💪。

现在大多数公司招人看中项目经验、工作经验,我也承认这些也确实非常重要。但是我时常会想,是否有一条捷径,可以让我们更快的强大起来、更快的走向成功?我也不确定,我也在摸索着前行...但我相信,站在巨人们的肩旁上,借鉴他们探索出的东西,加上自己的那一份努力,至少可以让自己少走一些弯路。

为什么会写这篇程序优化的总结,主要是面向如今这个大数据驱动的时代。如果没有大批量的数据让你去处理、如果没有大量业务需要你去书写,那么有必要去在乎那零点几毫秒的优化或是复用那些代码吗?

1. 选用恰当的数据结构

去重操作是我们日常会遇到的一个问题。我无数次遇到去重的问题的时候,可能大家想的都一样: 卧槽,也太简单了,给我两秒钟...然后做了如下操作:

ArrayList<String> arr = new ArrayList<String>(Arrays.asList(......)); //待去重数组 

ArrayList<String> arrTemp = new ArrayList<String>();
for(String str: arr) {
  if (!arrTemp.contains(str)) {
    arrTemp.add(str);
    // 处理操作
  }
}

这是正常人的思路吧,使用一个临时数组来存放新的元素,然后便利的时候判断是否包含。嗯...功能实现了,但是这个待遍历的数组变成了几十万条数据...然后我按下了 run 跑程序并等待程序的结束,然后我在程序finish 之前陷入了长时间的沉思......然后到处找呀找,查阅资料和博客,最后发现了一种效率非常赞的数据结构 - HashSet, 我非常激动,MD太辛苦了找了辣磨久终于找到好的办法了,这个时候之前的程序运行结束了......

先想想为什么这种数组的方式会很慢。在我们每一次循环中,我们都会使用contains()去遍历整个arrTemp,随着数组越来越长,速度会越来越慢。接下来看看 HashSet 的存储结构示意图:

HashSet存储结构.jpg

  在HashSet中,每一个key都对应着一个hashcode,因此通过计算可以先找到是在数组中 0~15 的哪一个位置,然后再去寻找这条链上有没有这个对应的key值。这么一来在速率上明显快了不上。

ArrayList<String> arr = new ArrayList<String>(Arrays.asList(......)); //待去重数组 

HashSet set = new HashSet();

for (String str : arr) {
  if (set.add(str)) {
        // 没有重复元素
  } else {
       // 存在重复元素...
  }
}

换了一种数据结构,在程序的执行效率上有了很大的改观。所以说,选择一种好的数据结构是很重要的~

2. 养成复用对象的好习惯

在程序的运行过程中,JVM堆上存放着活对象和程序无法再次引用的垃圾对象,当堆空间的空闲部分低于某个比例或者无法满足为新对象分配空间时,JVM都会触发一次垃圾收集(Garbage Collect,GC),触发gc时,JVM会暂停所有的用户线程。gc触发太过频繁或是时间过长,都会导致程序性能大大下降。
所以说,遇到此类问题我们应该怎么办呢? 我们需要尽量去重用对象,以减少gc的触发几率。比如说像这样:

String str = "";
for(...) {
  str = ....;
  str ......
}

顺便说一下,不只是对象的创建,循环里面最好别放一些没必要的操作,特别是耗时的操作。

3. 提高字符串拼接的效率

字符串的拼接是我们写程序的时候最最常见的情况了,我相信大家肯定都遇到过,并且以后也会继续和它打交道。对于文件IO读写而言,大家一定对于这句话并不陌生:

str += stringBuffer.readLine() + "\n";

我们在读取文件的我时候,经常会按行来读取,然后将其内容进行依次追加,最后将内容结果写出。

如果在循环次数不多的情况下,并且对效率的要求不高的话,为了图方便这么些没问题。但是,如果循环次数非常多的话,这种方式做字符串的拼接效率实在太低了。这里有一种更好的方式:使用 StringBuilder ,我们来看一个效率的对比结果吧。

首先我们假设有一个100000次的循环,每一次里面都会将内容"aaa123"追加到结果中。

   long start = System.currentTimeMillis();
        int size = 100000;
        String result = "";
        for (int i = 0; i < size; i++) {
            result += "aaa123";
        }
        long end = System.currentTimeMillis();
        System.out.println(end - start + "ms");
string追加.png

然后再来看一下改成StringBuilder之后的代码和运行结果:

  long start = System.currentTimeMillis();
        int size = 100000;
        String result = "";
        StringBuilder builder = new StringBuilder();
        for (int i = 0; i < size; i++) {
            builder.append("aaa123");
        }
        result = builder.toString();
        long end = System.currentTimeMillis();
        System.out.println(end - start + "ms");
StringBuilder追加.png

这效率简直高了几个数量级了...为什么StringBuilder可以做到在这样多的循环次下达到这么高的效率呢?我大致解释下...

对于使用'' + ''号来追加的方式而言,String对象在赋值以后是没法改变的,所以这种使用'' + ''号的追加原理其实每一次都重新创建了一个String 对象,然后对其赋值。这样不断新建对象,就像之前我们所讲的那样,会频繁触发gc,而且会随着内容越来越多使得创建String对象的速度越来越慢。

然而对于StringBuilderappend追加方式,是将string 转换为一个char[] 数组,然后通过扩展数组的方式将追加内容放入原始数组中。

4. for循环的正确使用姿势

对于for循环的使用,我的建议是:

  • 遍历数组使用基于索引的循环方式
  • 遍历集合使用基于迭代器的循环方式

具体的怎么去使用,我在这里就不做过多说明了。
主要讲讲怎么进行循环的优化吧。

(1) ''外小内大''原则
 for (int i = 0; i < 1000000 ; i++ ) {
            for (int j = 0; j < 1000 ; j++ ) {
                for (int k = 0 ; k < 10; k++) {

                }
            }
        }
循环1.png

这是一个典型的循环嵌套的结构,但是呢for循环应该是要满足一个''外小内大''的模式。这是我们改良后的代码:

 for (int i = 0; i < 10 ; i++ ) {
            for (int j = 0; j < 1000 ; j++ ) {
                for (int k = 0 ; k < 1000000; k++) {

                }
            }
        }
循环2.png

结果很明显,运行效率上面快了不少。但是这还不是最完美的做法~

        int i,j,k;
        for (i = 0; i < 10 ; i++ ) {
            for (j = 0; j < 1000 ; j++ ) {
                for (k = 0 ; k < 1000000; k++) {

                }
            }
        }

嗯,这样差不多就比较完美了😄

(2)避免没必要的操作

还是先来看一段原始代码:

for(int i = 0; i < list.size(); i++) {
     i = i * a * b;
}

这段代码有两个问题:第一个问题是每次循环都会进行一次 list.size()的计算,第二个问题是每次都会进行 a * b的计算。这些都是只需要进行一次的操作,没必要每次都进行。

优化后的代码如下:

int c = a * b;
int len = list.size();
for(int i = 0; i < len; i++) {
    i = i * c;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容