广告
返回顶部
首页 > 资讯 > 前端开发 > html >Javascript中怎么实现函数声明与递归调用
  • 456
分享到

Javascript中怎么实现函数声明与递归调用

2024-04-02 19:04:59 456人浏览 安东尼
摘要

javascript中怎么实现函数声明与递归调用,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。1、函数声明变量式声明先创建一个匿名函数,然后把

javascript中怎么实现函数声明与递归调用,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

1、函数声明

变量式声明先创建一个匿名函数,然后把它赋值给一个指定的变量:

var f = function () { // function body };

通常我们不必关心等号右边表达式的作用域是全局还是某个闭包内,因为它只能通过等号左边的变量f来引用,应该关注的是变量f的作用域。如果f指向函数的引用被破坏(f = null),且函数没有被赋值给任何其它变量或对象属性,匿名函数会因为失去所有引用而被垃圾回收机制销毁。

也可以使用函数表达式创建函数:

function f() { // function body }

与变量式不同的是,这种声明方式会为函数的一个内置属性name赋值。同时把函数赋值给当前作用域的一个同名变量。(函数的name属性,configurable、enumerable和writable均为false)

function f() { // function body }  console.log(f.name); // "f"  console.log(f); // f()

Javascript变量有一个的特别之处,就是会把变量的声明提前,表达式式的函数声明,也会把整个函数的定义前置,因此你可以在函数定义之前使用它:

console.log(f.name); // "f"  console.log(f); // f()  function f() { // function body }

函数表达式的声明会被提升到作用域顶层,试试下面的代码,它们不是本文的重点:

var a = 0;  console.log(a); // 0 or a()?  function a () {}

Crockford建议永远使用***种方式声明函数,他认为第二种方式放宽了函数必须先声明后使用的要求从而会导致混乱。(Crockford是一个类似于罗素口中用来比喻维特根斯坦的"有良心的艺术家"那样的"有良心的程序员",这句话很拗口吧)

函数式声明

function f() {}

看起来是

var f = function f(){};

的简写。

var a = function b(){};

的表达式,创建一个函数并把内置的name属性赋值为"b",然后把这个函数赋值给变量a,你可以在外部使用a()来调用它,但却不能使用b(),因为函数已被赋值给a,所以不会再自动创建一个变量b,除非你使用var b = a声明一个变量b。当然这个函数的name是"b"而不是"a"。

使用Function构造函数也可用来创建函数:

var f = new Function("a,b,c","return a+b+c;");

这种方式其实是在全局作用域内生成一个匿名函数,并把它赋值给变量f。

2、递归调用

递归被用来简化许多问题,这需要在一个函数体中调用它自己:

// 一个简单的阶乘函数  var f = function (x) {      if (x === 1) {          return 1;      } else {          return x * f(x - 1);      }  };

Javascript中函数的巨大灵活性,导致在递归时使用函数名遇到困难,对于上面的变量式声明,f是一个变量,所以它的值很容易被替换:

var fn = f;  f = function () {};

函数是个值,它被赋给fn,我们期待使用fn(5)可以计算出一个数值,但是由于函数内部依然引用的是变量f,于是它不能正常工作了。

函数式的声明看起来好些,但很可惜:

function f(x) {      if (x === 1) {          return 1;      } else {          return x * f(x - 1);      }  }  var fn = f;  f = function () {}; // may been warning by browser  fn(5); // NaN

看起来,一旦我们定义了一个递归函数,便须注意不要轻易改变变量的名字。

上面谈论的都是函数式调用,函数还有其它调用方式,比如当作对象方法调用。

我们常常这样声明对象:

var obj1 = {      num : 5,      fac : function (x) {          // function body      }  };

声明一个匿名函数并把它赋值给对象的属性(fac)。

如果我们想要在这里写一个递归,就要引用属性本身:

var obj1 = {      num : 5,      fac : function (x) {          if (x === 1) {              return 1;          } else {              return x * obj1.fac(x - 1);          }      }  };

当然,它也会遭遇和函数调用方式一样的问题:

var obj2 = {fac: obj1.fac};  obj1 = {};  obj2.fac(5); // Sadness

方法被赋值给obj2的fac属性后,内部依然要引用obj1.fac,于是…失败了。

换一种方式会有所改进:

var obj1 = {       num : 5,       fac : function (x) {          if (x === 1) {              return 1;          } else {              return x * this.fac(x - 1);          }      }  };  var obj2 = {fac: obj1.fac};  obj1 = {};  obj2.fac(5); // ok

通过this关键字获取函数执行时的context中的属性,这样执行obj2.fac时,函数内部便会引用obj2的fac属性。

可是函数还可以被任意修改context来调用,那就是***的call和apply:

obj3 = {};  obj1.fac.call(obj3, 5); // dead again

于是递归函数又不能正常工作了。

我们应该试着解决这种问题,还记得前面提到的一种函数声明的方式吗?

var a = function b(){};

这种声明方式叫做内联函数(inline function),虽然在函数外没有声明变量b,但是在函数内部,是可以使用b()来调用自己的,于是

var fn = function f(x) {      // try if you write "var f = 0;" here      if (x === 1) {          return 1;      } else {          return x * f(x - 1);      }  };  var fn2 = fn;  fn = null;  fn2(5); // OK
// here show the difference between "var f = function f() {}" and "function f() {}"  var f = function f(x) {      if (x === 1) {          return 1;      } else {          return x * f(x - 1);      }  };  var fn2 = f;  f = null;  fn2(5); // OK
var obj1 = {      num : 5,      fac : function f(x) {          if (x === 1) {              return 1;          } else {              return x * f(x - 1);          }      }  };  var obj2 = {fac: obj1.fac};  obj1 = {};  obj2.fac(5); // ok   var obj3 = {};  obj1.fac.call(obj3, 5); // ok

就这样,我们有了一个可以在内部使用的名字,而不用担心递归函数被赋值给谁以及以何种方式被调用。

Javascript函数内部的arguments对象,有一个callee属性,指向的是函数本身。因此也可以使用arguments.callee在内部调用函数:

function f(x) {      if (x === 1) {          return 1;      } else {          return x * arguments.callee(x - 1);      }  }

但arguments.callee是一个已经准备被弃用的属性,很可能会在未来的ECMAscript版本中消失,在ECMAscript 5中"use strict"时,不能使用arguments.callee。

看完上述内容,你们掌握Javascript中怎么实现函数声明与递归调用的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注编程网html频道,感谢各位的阅读!

--结束END--

本文标题: Javascript中怎么实现函数声明与递归调用

本文链接: https://www.lsjlt.com/news/73187.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Javascript中怎么实现函数声明与递归调用
    Javascript中怎么实现函数声明与递归调用,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。1、函数声明变量式声明先创建一个匿名函数,然后把...
    99+
    2022-10-19
  • 怎么调用Python递归函数与匿名函数
    这篇文章主要讲解了“怎么调用Python递归函数与匿名函数”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么调用Python递归函数与匿名函数”吧!一. 递归函数  a) 引言:在一个函数中...
    99+
    2023-06-02
  • Javascript中怎么实现函数声明和函数表达式
    Javascript中怎么实现函数声明和函数表达式,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Javascript是一种由N...
    99+
    2022-10-19
  • Oracle递归函数怎么用java实现
    在Java中,你可以通过创建一个递归函数来实现Oracle递归。以下是一个使用Java实现Oracle递归的示例:```javapu...
    99+
    2023-09-26
    Oracle java
  • JavaScript中如何使用函数递归实现数组扁平化
    小编给大家分享一下JavaScript中如何使用函数递归实现数组扁平化,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!数组扁平化是...
    99+
    2022-10-19
  • MySQL中怎么实现存储过程递归调用
    今天就跟大家聊聊有关MySQL中怎么实现存储过程递归调用,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。 有分类表tb_system_cat...
    99+
    2022-10-18
  • Java递归调用怎么实现数字的逆序输出
    这篇“Java递归调用怎么实现数字的逆序输出”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java递归调用怎么实现数字的逆序...
    99+
    2023-07-06
  • 怎么用Java数据结构与算法实现递归与回溯
    这篇文章主要介绍“怎么用Java数据结构与算法实现递归与回溯”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“怎么用Java数据结构与算法实现递归与回溯”文章能帮助大家解决问题。1.什么是递归?简单的说...
    99+
    2023-06-29
  • C语言怎么运用函数的递归实现汉诺塔
    这篇文章主要讲解了“C语言怎么运用函数的递归实现汉诺塔”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C语言怎么运用函数的递归实现汉诺塔”吧!1、汉诺塔是如何实现的下面是有三个盘子的示例:从左...
    99+
    2023-07-02
  • Flex与JavaScript中怎么实现交互中调用
    今天就跟大家聊聊有关Flex与JavaScript中怎么实现交互中调用,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。一、在JavaScript中调用Flex方法在Flex中可以用Ex...
    99+
    2023-06-17
  • 怎么在MySQL中通过自定义函数实现递归查询父级ID或者子级ID
    这篇文章给大家介绍怎么在MySQL中通过自定义函数实现递归查询父级ID或者子级ID,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一、环境准备1、建表CREATE TABLE&...
    99+
    2022-10-18
  • JavaScript中怎么利用sleep函数实现休眠或等待功能
    JavaScript中怎么利用sleep函数实现休眠或等待功能,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。Ja使JavaSc...
    99+
    2022-10-19
  • 微信小程序Page中怎么实现data数据操作和函数调用方法
    这篇文章主要为大家展示了“微信小程序Page中怎么实现data数据操作和函数调用方法”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“微信小程序Page中怎么实现d...
    99+
    2022-10-19
软考高级职称资格查询
推荐阅读
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作