3

**在 python 中,我可以通过(double-splat) 运算符传递其键与参数名称匹配的字典:

def foo(a, b):
    print (a - b)

args = {'b': 7, 'a': 10}

foo(**args) # prints 3

如何在 ES6 中做同样的事情?这不起作用:

function foo(a, b) {
    console.log(a - b)
}

args = {b: 7, a: 10}

foo(...args)

注意:我正在寻找一种不涉及更改签名的解决方案,foo因为我希望它以任何一种方式使用(有和没有解构)。所以以下应该工作:

foo(<magic>args);

foo(123, 456);

奖励问题:为什么错误消息“未定义不是函数”?这里到底是什么未定义?

(正如@Nina Scholz 在评论中回答的那样,这是因为...需要它的参数 have Symbol.iterator,这不是为对象定义的)。

4

2 回答 2

4

如何在 ES6 中做同样的事情?

JS 中没有命名参数,只有位置参数。所以答案是:你不能。

正如@Andy 建议的那样,您可以做的是通过对象传递来模拟命名参数。

function foo({ a, b }) {
    console.log(a - b);
}

let args = { b: 7, a: 10 };

foo(args);

或者你可以做成args一个数组,这样你就可以将它分解为位置参数。

function foo(a, b) {
    console.log(a - b);
}

let args = [10, 7];

foo(...args);

好的,好的,只是为了论证:可以编写一个函数,以所需的顺序提取参数foo并产生属性。args

function * yolo(args, fn) {
    const names = fn.toString().match(/\(.+\)/)[0]
                    .slice(1, -1).split(',')
                    .map(x => x.trim());

    while (names.length) {
        yield args[names.shift()];
    }
}

function foo(a, b) {
    console.log(a - b);
}

const args = { b: 7, a: 10 };

foo(...yolo(args, foo));

我不敢在生产中使用它。

于 2017-08-17T13:33:57.090 回答
1

您需要args用花括号括起来,并再次将其括在函数的参数列表中。

function foo({a, b}) {
    console.log(a - b)
}

let args = {b: 7, a: 10}

foo({...args})
于 2017-08-17T12:51:06.873 回答