递归地调用 javascript 函数

我可以在变量中创建一个递归函数,如下所示:

/* Count down to 0 recursively.
*/
var functionHolder = function (counter) {
output(counter);
if (counter > 0) {
functionHolder(counter-1);
}
}

这样,functionHolder(3);将输出 3 2 1 0:

var copyFunction = functionHolder;

copyFunction(3);会像上面一样输出 3 2 1 0。如果我改变 functionHolder如下:

functionHolder = function(whatever) {
output("Stop counting!");

然后 functionHolder(3);会给 Stop counting!,正如预期的那样。

copyFunction(3);现在给出的是 3 Stop counting!,因为它指向的是 functionHolder,而不是函数(它自己指向的函数)。这在某些情况下是可取的,但是有没有一种方法来编写函数,使它调用自己而不是保存它的变量?

也就是说,是否有可能改变 只有的线 functionHolder(counter-1);,以便通过所有这些步骤仍然给 3 2 1 0当我们调用 copyFunction(3);?我尝试了 this(counter-1);,但它给我的错误 this is not a function

152857 次浏览

Using Named Function Expressions:

You can give a function expression a name that is actually private and is only visible from inside of the function ifself:

var factorial = function myself (n) {
if (n <= 1) {
return 1;
}
return n * myself(n-1);
}
typeof myself === 'undefined'

Here myself is visible only inside of the function itself.

You can use this private name to call the function recursively.

See 13. Function Definition of the ECMAScript 5 spec:

The Identifier in a FunctionExpression can be referenced from inside the FunctionExpression's FunctionBody to allow the function to call itself recursively. However, unlike in a FunctionDeclaration, the Identifier in a FunctionExpression cannot be referenced from and does not affect the scope enclosing the FunctionExpression.

Please note that Internet Explorer up to version 8 doesn't behave correctly as the name is actually visible in the enclosing variable environment, and it references a duplicate of the actual function (see patrick dw's comment below).

Using arguments.callee:

Alternatively you could use arguments.callee to refer to the current function:

var factorial = function (n) {
if (n <= 1) {
return 1;
}
return n * arguments.callee(n-1);
}

The 5th edition of ECMAScript forbids use of arguments.callee() in strict mode, however:

(From MDN): In normal code arguments.callee refers to the enclosing function. This use case is weak: simply name the enclosing function! Moreover, arguments.callee substantially hinders optimizations like inlining functions, because it must be made possible to provide a reference to the un-inlined function if arguments.callee is accessed. arguments.callee for strict mode functions is a non-deletable property which throws when set or retrieved.

You can access the function itself using arguments.callee [MDN]:

if (counter>0) {
arguments.callee(counter-1);
}

This will break in strict mode, however.

Here's one very simple example:

var counter = 0;


function getSlug(tokens) {
var slug = '';


if (!!tokens.length) {
slug = tokens.shift();
slug = slug.toLowerCase();
slug += getSlug(tokens);


counter += 1;
console.log('THE SLUG ELEMENT IS: %s, counter is: %s', slug, counter);
}


return slug;
}


var mySlug = getSlug(['This', 'Is', 'My', 'Slug']);
console.log('THE SLUG IS: %s', mySlug);

Notice that the counter counts "backwards" in regards to what slug's value is. This is because of the position at which we are logging these values, as the function recurs before logging -- so, we essentially keep nesting deeper and deeper into the call-stack before logging takes place.

Once the recursion meets the final call-stack item, it trampolines "out" of the function calls, whereas, the first increment of counter occurs inside of the last nested call.

I know this is not a "fix" on the Questioner's code, but given the title I thought I'd generically exemplify Recursion for a better understanding of recursion, outright.

I know this is an old question, but I thought I'd present one more solution that could be used if you'd like to avoid using named function expressions. (Not saying you should or should not avoid them, just presenting another solution)

  var fn = (function() {
var innerFn = function(counter) {
console.log(counter);


if(counter > 0) {
innerFn(counter-1);
}
};


return innerFn;
})();


console.log("running fn");
fn(3);


var copyFn = fn;


console.log("running copyFn");
copyFn(3);


fn = function() { console.log("done"); };


console.log("fn after reassignment");
fn(3);


console.log("copyFn after reassignment of fn");
copyFn(3);

You can use the Y-combinator: (Wikipedia)

// ES5 syntax
var Y = function Y(a) {
return (function (a) {
return a(a);
})(function (b) {
return a(function (a) {
return b(b)(a);
});
});
};


// ES6 syntax
const Y = a=>(a=>a(a))(b=>a(a=>b(b)(a)));


// If the function accepts more than one parameter:
const Y = a=>(a=>a(a))(b=>a((...a)=>b(b)(...a)));

And you can use it as this:

// ES5
var fn = Y(function(fn) {
return function(counter) {
console.log(counter);
if (counter > 0) {
fn(counter - 1);
}
}
});


// ES6
const fn = Y(fn => counter => {
console.log(counter);
if (counter > 0) {
fn(counter - 1);
}
});

Using filter and map, recursion example removing null properties from an object

const obj = {
name: {
first: "Jeson",
middle: null,
last: "Holder"
},
age: 45
}
function removeNullOrEmpty(obj){
return Object.fromEntries(
Object.entries(obj)
.filter(([_, v])=> v!== null && v.length !== 0)
.map(([k, v])=>[k, v === Object(v)?removeNullOrEmpty(v):v])
)
}


console.log(removeNullOrEmpty(obj))