编译器优化之Loop Unrolling

前言

开发过程中,我们有可能会写段循环语句来执行一段代码,完成一个功能点。运行没问题看到结果后,一切万事大吉,功能完成了就可以了。但对于这段代码背后是如何执行的,很少有人愿意去深究,除非涉及到一些性能问题,不得不改进循环时才会去真正的探究本质。其实对于我们写的循环语句,编译器有可能帮助我们做了一定程度上的优化。这种优化的技术,就叫做Loop Unrolling 。

Loop Unrolling (循环展开)

循环展开,是一种牺牲程序的大小来加快程序执行速度的优化方法。可以由程序员完成,也可由编译器自动优化完成。循环展开最常用来降低循环开销,为具有多个功能单元的处理器提供指令级并行。也有利于指令流水线的调度。

本文主要探讨编译器对于Loop Unrolling优化问题。

有如下代码:

c++ int loop(int i) {     for(;i<10;i++);     return i; }

对于聪明的人类来说,看一下逻辑就可以得到结果,如果参数i小于等于10,则返回10;否则返回i(即一次比较就知道结果)。

但对于编译器来说,在编译代码时会有自己的一套规则。当编辑优化参数设置为-o0时,编译器会按部就班的来执行代码逻辑。

即会按照流程执行for语句,然后再返回结果。如下是汇编结果,

不优化.png

可以看到将值与9比较然后决定是返回循环还是结束循环。

那有没有聪明一点的做法呢?答案是肯定的,在编译器优化选项中设置-o2,编译器会对函数做优化,执行Loop Unrolling,函数会被优化成一条简单的if指令,如下是汇编结果,

优化.png

可以看到,直接跟10比较来返回结果。

编译器将一个循环转变为不需要循环的过程,这个过程就叫做Loop Unrolling。

有关汇编指令的问题可以去简单的复习一下汇编。

总结

Loop Unrolling 有助于提升App的性能,但也有相应的缺点。它的优缺点如下:

  • 优点

    • 分支预测失败减少。

    • 如果循环結構内语句没有数据相关,增加了并发执行的机会。

    • 可以在执行时动态循环展开。这种情况在编译时也不可能掌握。

  • 缺点

    • 代码膨胀

    • 代码可读性降低,除非是编译器透明执行循环展开。

    • 循环結構内包含递归可能会降低循环展开的效益。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值