C语言技巧:有if时使用likely和unlikely让代码运行更快

文章介绍了在单片机和嵌入式编程中,如何利用likely和unlikely这两个宏进行if-else分支的优化,通过告知编译器条件发生的概率,调整代码布局以减少跳转开销,提高CPU的指令预取效率,从而提升程序运行速度。
摘要由CSDN通过智能技术生成

目录

1.likely和unlikely

2. 注意​

3. 原理


在单片机/嵌入式编程中,对速度要求比较高,likely和unlikely就是一个比较好的技巧,适用于有if-else分支,且知道哪个发生概率大的情况​。

参考文章:C语言技巧:有if时使用likely和unlikely让代码运行更快

1.likely和unlikely

这是两个宏,当有if-else分支时,告诉编译器,哪个条件更加有可能发生。对​if-else分支结构进行优化。likely代表if分支大概率会发生,unlikely代表if分支大概率不会发生。

 #define likely(x) __builtin_expect(!!(x), 1)
 #define unlikely(x) __builtin_expect(!!(x), 0)
__builtin_expect是编译器内建函数,原型为long __builtin_expect (long exp, long c)。

​题外话: !!是C语言中处理逻辑表达式的一个技巧。因为C语言中没有布尔变量,所以布尔值是用整形来代替的,0为假,非0为真。当x为0时,!(x)为1,!!(x)为0,!!的运算没有什么意义;但当x为非0时(比如100),!(x)为0,!!(x)为1,这样就达到了将非0值(比如100)全部都映射为1的效果。

例子:

​假如有这样一个函数:

int fun(int a)
{
  if(a>100)
  ......
}
在实际的运行情况中,a>100不太可能发生。那就这样写:

int fun(int a)
{
  if(unlikely(a>100))
  ......
}
同理,如果实际情况是a>100的情况更多,就用​likely。

2. 注意​

1.likely和unlikely使用判断必须准确,写反会使运行速度变慢。

2.编译时优化一般需要至少使用-O2选项,否则优化不起作用。

3.能否使用与编译器有关,gcc可以,clang好像也可以,msvc​好像不可以。

3. 原理

编译器在编译生成汇编代码时会在编译选项的引导下调整if分支内代码的位置,如果是likely修饰过的就调整到前面,如果是unlikely修饰过的就调整到后面。放到前面的代码可以节省跳转指令带来的时间开销,从而达到提升效率的目的。

现在的CPU都有ICache和流水线机制。在运行当前这条指令时,ICache会预取后面的指令,以提升运行效率。但是如果条件分支不满足,也就会跳转到了其他指令,那预取的指令也就没有用了,这样就降低了流水线的效率。

如果使用likely和unlikely来告诉编译器,哪个更加有可能发生,编译器将大概率执行的代码放在靠前的位置,就可以大大提高指令预取值的命中率,从而达到提升效率的目的。

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值