免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
12下一页
最近访问板块 发新帖
查看: 11827 | 回复: 11
打印 上一主题 下一主题

[C] 尾递归真的会被优化吗? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2014-01-23 13:27 |只看该作者 |倒序浏览
本帖最后由 linyunxian 于 2014-01-23 16:24 编辑

一直听说尾递归的效率高是因为它会被编译器优化,不会有产生调用栈的开销。
可从调试来看,却没发现我普通的递归有什么差别。
下面是尾递归的测试实例。

  1. long long fact_tr_aux(long long n, long long *result)
  2. {
  3.         printf("%x\n", &n); /* print &n in stack */

  4.         if (n==0 || n==1)
  5.                 return *result;

  6.         *result *= n;

  7.         return fact_tr_aux(n-1, result);
  8. }

  9. long long fact_tr(long long n)
  10. {
  11.         long long result = 1;

  12.         return fact_tr_aux(n, &result);
  13. }

  14. int main(int argc, char *argv[])
  15. {
  16.         printf("%lld\n", fact_tr(atoi(argv[1])));

  17.         return 0;
  18. }
复制代码
无论从汇编还是gdb看调用栈,都发现一样的在栈上分配递归了。
这是为什么?

谢谢各位的回复。
总结一下:gcc也可能会对尾递归优化,前提是递归栈里没有变量需要保存。
注释掉这行似乎就可以:        printf("%x\n", &n); /* print &n in stack */
gcc不能判断&n是否属于上个栈帧上,所以没有优化。

论坛徽章:
0
2 [报告]
发表于 2014-01-23 13:28 |只看该作者
@humjb_1983 兄台解释一下?

论坛徽章:
1
天秤座
日期:2014-04-27 07:42:20
3 [报告]
发表于 2014-01-23 13:47 |只看该作者
编译器不会优化的,递归的效率主要取决于边界条件是否最优。

论坛徽章:
15
射手座
日期:2014-02-26 13:45:082015年迎新春徽章
日期:2015-03-04 09:54:452015年辞旧岁徽章
日期:2015-03-03 16:54:15羊年新春福章
日期:2015-02-26 08:47:552015年亚洲杯之卡塔尔
日期:2015-02-03 08:33:45射手座
日期:2014-12-31 08:36:51水瓶座
日期:2014-06-04 08:33:52天蝎座
日期:2014-05-14 14:30:41天秤座
日期:2014-04-21 08:37:08处女座
日期:2014-04-18 16:57:05戌狗
日期:2014-04-04 12:21:33技术图书徽章
日期:2014-03-25 09:00:29
4 [报告]
发表于 2014-01-23 13:54 |只看该作者
linyunxian 发表于 2014-01-23 13:27
一直听说尾递归的效率高是因为它会被编译器优化,不会有产生调用栈的开销。
可从调试来看,却没发现我普通 ...

据了解,编译器(至少是gcc)是不会去递归的,没有这个功能,当然肯定还会压栈,并带有栈溢出的风险,不知兄台是从哪儿看到编译器会做这样的优化的呢?呵呵

论坛徽章:
14
巨蟹座
日期:2013-11-19 14:09:4615-16赛季CBA联赛之青岛
日期:2016-07-05 12:36:0515-16赛季CBA联赛之广东
日期:2016-06-29 11:45:542015亚冠之全北现代
日期:2015-07-22 08:09:472015年辞旧岁徽章
日期:2015-03-03 16:54:15巨蟹座
日期:2014-12-29 08:22:29射手座
日期:2014-12-05 08:20:39狮子座
日期:2014-11-05 12:33:52寅虎
日期:2014-08-13 09:01:31巳蛇
日期:2014-06-16 16:29:52技术图书徽章
日期:2014-04-15 08:44:01天蝎座
日期:2014-03-11 13:06:45
5 [报告]
发表于 2014-01-23 13:56 |只看该作者
C/C++ 没有规定过 必须优化尾递归。事实上,也没有哪个编译器去优化尾递归。当然,如果有编译器去优化尾递归,也是符合标准的。

估计你说的是Java等,Java等才有“尾递归”的概念。

尾递归 很容易改写成 循环,所以C/C++是不会加入“尾递归”这个概念的,优化让编码者自己来。

论坛徽章:
15
射手座
日期:2014-02-26 13:45:082015年迎新春徽章
日期:2015-03-04 09:54:452015年辞旧岁徽章
日期:2015-03-03 16:54:15羊年新春福章
日期:2015-02-26 08:47:552015年亚洲杯之卡塔尔
日期:2015-02-03 08:33:45射手座
日期:2014-12-31 08:36:51水瓶座
日期:2014-06-04 08:33:52天蝎座
日期:2014-05-14 14:30:41天秤座
日期:2014-04-21 08:37:08处女座
日期:2014-04-18 16:57:05戌狗
日期:2014-04-04 12:21:33技术图书徽章
日期:2014-03-25 09:00:29
6 [报告]
发表于 2014-01-23 14:15 |只看该作者
linyunxian 发表于 2014-01-23 13:27
一直听说尾递归的效率高是因为它会被编译器优化,不会有产生调用栈的开销。
可从调试来看,却没发现我普通 ...

咨询了一下专家,确实对于“尾递归”的情况,也就是说函数体中用到的变量不需要栈保存,gcc -O2是可以进行优化的,会将其展开成循环,但如果递归函数中有分支就不行,比如路径遍历的实现。有分支就要保存条件变量,就需要压栈,这种情况就展不开。实际编码中绝大多数都是这种情况。

论坛徽章:
0
7 [报告]
发表于 2014-01-23 14:28 |只看该作者
回复 6# humjb_1983


    嗯!刚测试了一下!gcc编译c代码如果递归代码里的变量不需要在栈里维护,就有可能被优化掉。不产生新的栈桢。

论坛徽章:
8
CU大牛徽章
日期:2013-04-17 10:59:39CU大牛徽章
日期:2013-04-17 11:01:45CU大牛徽章
日期:2013-04-17 11:02:15CU大牛徽章
日期:2013-04-17 11:02:36CU大牛徽章
日期:2013-04-17 11:02:58技术图书徽章
日期:2013-12-04 10:48:50酉鸡
日期:2014-01-03 10:32:30辰龙
日期:2014-03-06 15:04:07
8 [报告]
发表于 2014-01-23 15:56 |只看该作者

论坛徽章:
0
9 [报告]
发表于 2014-01-23 16:22 |只看该作者
回复 8# shan_ghost


    gcc 的优化选项 ~-O1 -foptimize-sibling-calls~  或 ~-O2~ 都会使能尾递归的优化。
不过这种优化是有前提的,就是递归函数本身不依赖函数栈这种结构。

论坛徽章:
2
青铜圣斗士
日期:2015-11-26 06:15:59数据库技术版块每日发帖之星
日期:2016-07-24 06:20:00
10 [报告]
发表于 2014-01-24 13:56 |只看该作者

  1. printf("%x\n", &n); /* print &n in stack */
复制代码
这让编译器情何以堪。。。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP