免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
楼主: linyunxian
打印 上一主题 下一主题

[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 兄台解释一下?

论坛徽章:
0
3 [报告]
发表于 2014-01-23 14:28 |显示全部楼层
回复 6# humjb_1983


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

论坛徽章:
0
4 [报告]
发表于 2014-01-23 16:22 |显示全部楼层
回复 8# shan_ghost


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

论坛徽章:
0
5 [报告]
发表于 2014-01-24 14:37 |显示全部楼层
回复 10# OwnWaterloo


    是的,这条语句阻止了gcc的优化。但逻辑上来说,似乎它仍可以做到。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP