Стиль JavaScript для дополнительных обратных вызовов
У меня есть некоторые функции, которые иногда (не всегда) получают обратный вызов и запускают его. Является ли проверка, определен ли обратный вызов / функция, является хорошим стилем или есть лучший способ?
Пример:
function save (callback){
.....do stuff......
if(typeof callback !== 'undefined'){
callback();
};
};
10 ответов
Я лично предпочитаю
typeof callback === 'function' && callback();
typeof
Однако команда хитрая и должна использоваться только для "undefined"
а также "function"
Проблемы с typeof !== undefined
является то, что пользователь может передать определенное значение, а не функцию
Вы также можете сделать:
var noop = function(){}; // do nothing.
function save (callback){
callback = callback || noop;
.....do stuff......
};
Это особенно полезно, если вы используете callback
в нескольких местах.
Кроме того, если вы используете jQuery
, у вас уже есть такая функция, она называется $.noop
Просто делай
if (callback) callback();
Я предпочитаю вызывать обратный вызов, если он указан, независимо от его типа. Не позволяйте этому молча провалиться, поэтому разработчик знает, что он передал неверный аргумент, и может это исправить.
ECMAScript 6
// @param callback Default value is a noop fn.
function save(callback = ()=>{}) {
// do stuff...
callback();
}
Вместо того, чтобы сделать обратный вызов необязательным, просто назначьте значение по умолчанию и вызовите его независимо от того, что
const identity = x =>
x
const save (..., callback = identity) {
// ...
return callback (...)
}
Когда используется
save (...) // callback has no effect
save (..., console.log) // console.log is used as callback
Такой стиль называется продолжением стиля прохождения. Вот реальный пример, 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']))
// [ []
// , [ 'C' ]
// , [ 'B' ]
// , [ 'B', 'C' ]
// , [ 'A' ]
// , [ 'A', 'C' ]
// , [ 'A', 'B' ]
// , [ 'A', 'B', 'C' ]
// ]
Так как combinations
определен в стиле передачи продолжения, вышеупомянутый вызов фактически такой же
combinations (['A', 'B', 'C'], console.log)
// [ []
// , [ 'C' ]
// , [ 'B' ]
// , [ 'B', 'C' ]
// , [ 'A' ]
// , [ 'A', 'C' ]
// , [ 'A', 'B' ]
// , [ 'A', 'B', 'C' ]
// ]
Мы также можем передать пользовательское продолжение, которое делает что-то еще с результатом
console.log (combinations (['A', 'B', 'C'], combs => combs.length))
// 8
// (8 total combinations)
Стиль продолжения прохождения может использоваться с удивительно элегантными результатами
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)) // 55
// 55 is the 10th fibonacci number
// (0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, ...)
Я так устал видеть этот фрагмент снова и снова, я написал это:
var cb = function(g) {
if (g) {
var args = Array.prototype.slice.call(arguments);
args.shift();
g.apply(null, args);
}
};
У меня есть сотни функций, выполняющих такие вещи, как
cb(callback, { error : null }, [0, 3, 5], true);
или что угодно...
Я скептически отношусь ко всей стратегии "убедитесь, что она работает". Единственными законными ценностями являются функция или ложь. Если кто-то передает ненулевое число или непустую строку, что вы собираетесь делать? Как игнорирование проблемы решает ее?
Действительная функция основана на прототипе Function, используйте:
if (callback instanceof Function)
чтобы быть уверенным, что обратный вызов является функцией
Если критерии для выполнения обратного вызова таковы, определен он или нет, то все в порядке. Кроме того, я предлагаю проверить, действительно ли это функция в дополнение.
Я перешел на coffee-script и нашел аргументы по умолчанию - хороший способ решить эту проблему
doSomething = (arg1, arg2, callback = ()->)->
callback()
Это легко сделать с помощью ArgueJS:
function save (){
arguments = __({callback: [Function]})
.....do stuff......
if(arguments.callback){
callback();
};
};