详解详解Javascript函数声明与递归调用函数声明与递归调用
本篇文章详细的介绍了Javascript函数声明与递归调用,是JS入门学习中的基础知识,需要的朋友可以参考下。
Javascript的函数的声明方式和调用方式已经是令人厌倦的老生常谈了,但有些东西就是这样的,你来说一遍然后我再说一
遍。每次看到书上或博客里写的Javascript函数有四种调用方式,我就会想起孔乙己:茴字有四种写法,你造吗?
尽管缺陷有一堆,但Javascript还是令人着迷的。Javascript众多优美的特性的核心,是作为顶级对象(first-class objects)的
函数。函数就像其他普通对象一样被创建、被分配给变量、作为参数被传递、作为返回值以及持有属性和方法。函数作为顶级
对象,赋予了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、递归调用、递归调用
递归被用来简化许多问题,这需要在一个函数体中调用它自己:
评论0
最新资源