Обработка конкретных ошибок в JavaScript (подумайте об исключениях)


112

Как бы вы реализовали различные типы ошибок, чтобы вы могли выявлять определенные ошибки и позволять другим всплывать?

Один из способов добиться этого - изменить прототип Errorобъекта:

Error.prototype.sender = "";


function throwSpecificError()
{
    var e = new Error();

    e.sender = "specific";

    throw e;
}

Выявить конкретную ошибку:

try
{
    throwSpecificError();
}

catch (e)
{
    if (e.sender !== "specific") throw e;

    // handle specific error
}


У вас есть какие-нибудь альтернативы?

Ответы:


159

Чтобы создать собственные исключения, вы можете унаследовать от объекта Error:

function SpecificError () {

}

SpecificError.prototype = new Error();

// ...
try {
  throw new SpecificError;
} catch (e) {
  if (e instanceof SpecificError) {
   // specific error
  } else {
    throw e; // let others bubble up
  }
}

Минималистичный подход без наследования от Error может заключаться в создании простого объекта, имеющего имя и свойства сообщения:

function throwSpecificError() {
  throw {
    name: 'SpecificError',
    message: 'SpecificError occurred!'
  };
}


// ...
try {
  throwSpecificError();
} catch (e) {
  if (e.name == 'SpecificError') {
   // specific error
  } else {
    throw e; // let others bubble up
  }
}

2
При наследовании от Errorесть проблемы. См. Stackoverflow.com/questions/1382107/…
Crescent Fresh

5
проблема с этим кодом: в } catch (e) { if (e.name == 'SpecificError') { // specific error } else { throw e; // let others bubble up } }том, что он не будет работать в IE7, вызывая ошибку «Исключение выброшено и не обнаружено». Ниже приводится чрезвычайно глупое (как всегда) объяснение от msdn: «Вы включили оператор throw, но он не был заключен в блок try, или не было связанного блока catch для перехвата ошибки. Исключения возникают из блока try используя оператор throw, и пойманный за пределами блока try с помощью оператора catch ".
Евгений Кузьменко

1
Что ж, Microsoft C # определенно обрабатывает ошибки лучше, чем Javascript: P. Mozzilla добавила в Firefox что-то подобное. Хотя это не входит в стандарт Ecmascript, даже ES6, но они также объясняют, как сделать его совместимым, хотя это не так кратко. В основном то же, что и выше, но с использованием instanceOf. Проверьте здесь
Барт

В Javascript вы можете использовать все, что захотите, будь то простая строка, число (подумайте о коде ошибки) или полностью квалифицированный объект. Сладкий!
Авраам Брукс

1
@LuisNell, если вы внимательно посмотрите на мой пример кода, вы увидите, что я не предлагал использовать nameсвойство функции конструктора. Я предлагал бросить сделанный на заказ объект со nameсвойством, которое не сломается ...
CMS

15

Как отмечено в комментариях ниже, это специфично для Mozilla, но вы можете использовать блоки «условного перехвата». например:

try {
  ...
  throwSpecificError();
  ...
}
catch (e if e.sender === "specific") {
  specificHandler(e);
}
catch (e if e.sender === "unspecific") {
  unspecificHandler(e);
}
catch (e) {
  // don't know what to do
  throw e;
} 

Это дает нечто более похожее на типизированную обработку исключений, используемую в Java, по крайней мере, синтаксически.


Совместите с ответом CMS, и он идеален.
Атес Горал,

3
Об условном улове я либо раньше не знал, либо забыл. Спасибо за обучение / напоминание! +1
Атес Горал

12
Поддерживается только Firefox (начиная с версии 2.0). Он даже не выполняет синтаксический анализ в других браузерах; вы получаете только синтаксические ошибки.
Crescent Fresh

10
Да, это расширение только для Mozilla, оно даже не предлагается для стандартизации. Поскольку это функция уровня синтаксиса, ее невозможно нюхать и при желании использовать.
bobince

3
Кроме того, предлагаемое решение является нестандартным. Цитата из [Справочника Mozilla JavaScript [( developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/… ):This feature is non-standard and is not on a standards track. Do not use it on production sites facing the Web: it will not work for every user. There may also be large incompatibilities between implementations and the behavior may change in the future.
informatik01

10

try-catch-finally.js

Используя try-catch-finally.js , вы можете вызвать _tryфункцию с анонимным обратным вызовом, который она вызовет, и вы можете связать .catchвызовы для обнаружения определенных ошибок и .finallyвызов для выполнения в любом случае.

пример

_try(function () {
    throw 'My error';
})
.catch(Error, function (e) {
    console.log('Caught Error: ' + e);
})
.catch(String, function (e) {
    console.log('Caught String: ' + e);
})
.catch(function (e) {
    console.log('Caught other: ' + e);
})
.finally(function () {
    console.log('Error was caught explicitly');
});

Пример с современными стрелочными функциями и шаблонными литералами

_try(() => {
  throw 'My error';
}).catch(Error, e => {
  console.log(`Caught Error: ${e}`);
}).catch(String, e => {
  console.log(`Caught String: ${e}`);
}).catch(e => {
  console.log(`Caught other: ${e}`);
}).finally(() => {
  console.log('Error was caught explicitly');
});

2

Модуль для использования экспорта

/**
 * Custom InputError
 */
class InputError extends Error {
  /**
   * Create InputError
   * @param {String} message
   */
  constructor(message) {
    super(message);
    this.name = this.constructor.name;
    Error.captureStackTrace(this, this.constructor);
  }
}

/**
 * Custom AuthError
 */
class AuthError extends Error {
  /**
   * Create AuthError
   * @param {String} message
   */
  constructor(message) {
    super(message);
    this.name = this.constructor.name;
    Error.captureStackTrace(this, this.constructor);
  }
}

/**
 * Custom NotFoundError
 */
class NotFoundError extends Error {
  /**
   * Create NotFoundError
   * @param {String} message
   */
  constructor(message) {
    super(message);
    this.name = this.constructor.name;
    Error.captureStackTrace(this, this.constructor);
  }
}

module.exports = {
  InputError: InputError,
  AuthError: AuthError,
  NotFoundError: NotFoundError
};

Импортировать в скрипт:

const {InputError, AuthError, NotFoundError} = require(path.join(process.cwd(), 'lib', 'errors'));

Использование:

function doTheCheck = () =>
  checkInputData().then(() => {
    return Promise.resolve();
  }).catch(err => {
    return Promise.reject(new InputError(err));
  });
};

Телефонный код внешний:

doTheCheck.then(() => {
  res.send('Ok');
}).catch(err => {
  if (err instanceof NotFoundError) {
    res.status(404).send('Not found');
  } else if (err instanceof AuthError) {
    res.status(301).send('Not allowed');
  } else if (err instanceof InputError) {
    res.status(400).send('Input invalid');
  } else {
    console.error(err.toString());
    res.status(500).send('Server error');
  }
});

0

Мне не нравилось ни одно из этих решений, поэтому я сделал свое. Try-catch-finally.js довольно крутой, за исключением того, что если вы забудете один маленький символ подчеркивания (_) перед попыткой, тогда код все равно будет работать нормально, но ничего не поймают! Тьфу.

CatchFilter

Я добавил в свой код CatchFilter:

"use strict";

/**
 * This catches a specific error. If the error doesn't match the errorType class passed in, it is rethrown for a
 * different catch handler to handle.
 * @param errorType The class that should be caught
 * @param funcToCall The function to call if an error is thrown of this type
 * @return {Function} A function that can be given directly to the `.catch()` part of a promise.
 */
module.exports.catchOnly = function(errorType, funcToCall) {
  return (error) => {
    if(error instanceof errorType) {
      return funcToCall(error);
    } else {
      // Oops, it's not for us.
      throw error;
    }
  };
};

Теперь я могу фильтровать

Теперь я могу фильтровать как в C # или Java:

new Promise((resolve, reject => {
   <snip><snip>
}).catch(CatchFilter.catchOnly(MyError, err =>
   console.log("This is for my error");
}).catch(err => {
   console.log("This is for all of the other errors.");
});

-2
    <li>
      <span>onWarning:</span>
      <span id="msg_warning"></span>
    </li>

  try {
  // load face detection model
  await changeFaceDetector(MTCNN)
  changeInputSize(128)

  // try to access users webcam and stream the images
  // to the video element

    const stream = await navigator.mediaDevices.getUserMedia({ video: {} })
    const videoEl = $('#inputVideo').get(0)
    videoEl.srcObject = stream
  }
  catch(err) {
    //$("#msg_error").html(`Requested device not found`);
    $("#msg_error").html(err.message);
    console.log(err.message);
  }

Привет, добро пожаловать в StackOverflow. Как ваш ответ лучше / эффективнее / и т.д. из других 5 уже опубликованных ответов?
mjuarez
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.