Стиль JavaScript для дополнительных обратных вызовов - PullRequest
84 голосов
/ 22 июля 2011

У меня есть некоторые функции, которые иногда (не всегда) получают обратный вызов и запускают его. Является ли проверка, определен ли обратный вызов / функция, является хорошим стилем или есть лучший способ?

Пример:

function save (callback){
   .....do stuff......
   if(typeof callback !== 'undefined'){
     callback();
   };
};

Ответы [ 10 ]

129 голосов
/ 22 июля 2011

лично я предпочитаю

typeof callback === 'function' && callback();

Команда typeof является хитрой, однако ее следует использовать только для "undefined" и "function"

Проблема с typeof !== undefined состоит в том, что пользователь может передать определенное значение, а не является функцией

44 голосов
/ 22 июля 2011

Вы также можете сделать:

var noop = function(){}; // do nothing.

function save (callback){
   callback = callback || noop;
   .....do stuff......
};

Это особенно полезно, если вы используете callback в нескольких местах.

Кроме того, если вы используете jQuery, у вас уже есть такая функция, она называется $. Noop

33 голосов
/ 31 марта 2014

Просто сделай

if (callback) callback();

Я предпочитаю вызывать обратный вызов, если он указан, независимо от его типа. Не позволяйте, чтобы это молча провалилось, поэтому разработчик знает, что он передал неверный аргумент, и может это исправить.

7 голосов
/ 20 декабря 2017

ECMAScript 6

// @param callback Default value is a noop fn.
function save(callback = ()=>{}) {
   // do stuff...
   callback();
}
1 голос
/ 10 сентября 2018

Вместо того, чтобы сделать обратный вызов необязательным, просто назначьте значение по умолчанию и назовите его независимо от того, что

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, ...)
1 голос
/ 26 декабря 2014

Действительная функция основана на прототипе функции, используйте:

if (callback instanceof Function)

чтобы быть уверенным, что обратный вызов является функцией

1 голос
/ 22 июля 2011

Я так устал видеть один и тот же фрагмент снова и снова, я написал это:

  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);

или что-то еще ...

Я скептически отношусь ко всей стратегии "убедитесь, что она функционирует". Единственными законными значениями являются функция или ложь. Если кто-то передает ненулевое число или непустую строку, что вы собираетесь делать? Как игнорирование проблемы решает ее?

0 голосов
/ 12 апреля 2013

Это можно легко сделать с помощью ArgueJS :

function save (){
  arguments = __({callback: [Function]})
.....do stuff......
  if(arguments.callback){
    callback();
  };
};
0 голосов
/ 14 марта 2013

Я перешел на кофе-скрипт и нашел аргументы по умолчанию - хороший способ решить эту проблему

doSomething = (arg1, arg2, callback = ()->)->
    callback()
0 голосов
/ 22 июля 2011

Если критерии для выполнения обратного вызова таковы, определен он или нет, то все в порядке.Кроме того, я предлагаю проверить, действительно ли это функция в дополнение.

...