Вместо того, чтобы делать обратный вызов необязательным, просто назначьте значение по умолчанию и вызывайте его независимо от того, что
const identity = x =>
x
const save (..., callback = identity) {
return callback (...)
}
При использовании
save (...)
save (..., console.log)
Такой стиль называется продолжением передачи . Вот реальный пример, combinations
который генерирует все возможные комбинации ввода массива
const identity = x =>
x
const None =
Symbol ()
const combinations = ([ x = None, ...rest ], callback = identity) =>
x === None
? callback ([[]])
: combinations
( rest
, combs =>
callback (combs .concat (combs .map (c => [ x, ...c ])))
)
console.log (combinations (['A', 'B', 'C']))
Поскольку combinations
он определен в стиле передачи продолжения, приведенный выше вызов фактически такой же
combinations (['A', 'B', 'C'], console.log)
Мы также можем передать собственное продолжение, которое делает что-то еще с результатом
console.log (combinations (['A', 'B', 'C'], combs => combs.length))
Стиль передачи с продолжением можно использовать с удивительно элегантными результатами.
const first = (x, y) =>
x
const fibonacci = (n, callback = first) =>
n === 0
? callback (0, 1)
: fibonacci
( n - 1
, (a, b) => callback (b, a + b)
)
console.log (fibonacci (10))
typeof callback !== undefined
так что'