Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 709|回复: 0

Javascript中递归造成的堆栈溢出及解决方案

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-5-15 09:24:56 | 显示全部楼层 |阅读模式

    关于堆栈的溢出问题,在Javascript日常开发中很常见,Google了下,相关问题还是比较多的。本文旨在描述如何解决此类问题。 首先看一个实例(当然你可以使用更容易的方式实现,这里我们仅探讨递归):

    function isEven (num) {
        if (num === 0) {
            return true;
        }
    
        if (num === 1) {
            return false;
        }
    
        return isEven(Math.abs(num) - 2);
    }
    
    //Outputs: true
    console.log(isEven(10));
    
    //Outputs: false
    console.log(isEven(9));
    

    当我们把参数改成10000时,运行下例会发生堆栈溢出:

    function isEven (num) {
        if (num === 0) {
            return true;
        }
    
        if (num === 1) {
            return false;
        }
    
        return isEven(Math.abs(num) - 2);
    }
    
    //不同的javascript引擎报错可能不同
    //Outputs: Uncaught RangeError: Maximum call stack size exceeded 
    console.log(isEven(10000));
    

    原因是每次执行代码时,都会分配一定尺寸的栈空间(Windows系统中为1M),每次方法调用时都会在栈里储存一定信息(如参数、局部变量、返回值等等),这些信息再少也会占用一定空间,成千上万个此类空间累积起来,自然就超过线程的栈空间了。那么如何解决此类问题?

    使用闭包:

    function isEven (num) {
        if (num === 0) {
            return true;
        }
    
        if (num === 1) {
            return false;
        }
    
        return function() {
            return isEven(Math.abs(num) - 2);
        }
    }
    //Outputs: true
    console.log(isEven(4)()());
    

    此时每次调用时,返回一个匿名函数,匿名函数执行相关的参数和局部变量将会释放,不会额外增加堆栈大小。

    优化调用:

    上例调用比较麻烦,优化如下:

    function isEven (num) {
        if (num === 0) {
            return true;
        }
    
        if (num === 1) {
            return false;
        }
    
        return function() {
            return isEven(Math.abs(num) - 2);
        }
    }
    
    function trampoline (func, arg) {
        var value = func(arg);
    
        while(typeof value === "function") {
            value = value();
        }
    
        return value;
    }
    //Outputs: true
    console.log(trampoline(isEven, 10000));
    
    //Outputs: false
    console.log(trampoline(isEven, 10001));
    

    现在我们可以解决堆栈溢出问题了,但是不是感觉每次tarmpoline(isEven, 1000)这种调用方式不是很好,我们可以使用bind来绑定:

    function isEven(n) {
        /**
         * [isEvenInner 递归]
         * @param  {[type]}  num [description]
         * @return {Boolean}     [description]
         */
        function isEvenInner (n) {
            if (n === 0) {
                return true;
            }
    
            if (n === 1) {
                return false;
            }
    
            return function() {
                return isEvenInner(Math.abs(n) - 2);
            }
        }
        /**
         * [trampoline 迭代]
         * @param  {[type]} func [description]
         * @param  {[type]} arg  [description]
         * @return {[type]}      [description]
         */
        function trampoline (func, arg) {
            var value = func(arg);
    
            while(typeof value === "function") {
                value = value();
            }
    
            return value;
        }
    
        return trampoline.bind(null, isEvenInner)(n);
    }
    //Outputs: true
    console.log(isEven(10000));
    
    //Outputs: false
    console.log(isEven(10001));
    

    虽然上例实现了我们想要的效果,但是trampoline函数还是有一定的局限性:

    1.假设你只传递一个参数给递归函数

    value = func(arg); 修改为 value = func.apply(func, arg);

    2.假设最后的返回值不是一个函数 关于更健壮性的实现,请看underscore-contrib中源码。

    感谢您的阅读,文中不妥之处还望批评指正,文章已同步至个人博客如果你有好的建议,欢迎留言,么么哒!

    转载声明:

    本文标题:Javascript中递归造成的堆栈溢出及解决方案

    本文链接:http://www.zuojj.com/archives/1115.html,转载请注明转自Benjamin-专注前端开发和用户体验

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-2-3 12:58 , Processed in 0.060228 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表