Отключить предупреждения при загрузке не правильно сформированного HTML с помощью DomDocument (PHP) - PullRequest
73 голосов
/ 19 июля 2009

Мне нужно разобрать некоторые HTML-файлы, однако они не правильно сформированы и PHP выводит предупреждения. Я хочу избежать такого поведения отладки / предупреждения программно. Пожалуйста, порекомендуйте. Спасибо!

Код:

// create a DOM document and load the HTML data
$xmlDoc = new DomDocument;
// this dumps out the warnings
$xmlDoc->loadHTML($fetchResult);

Это:

@$xmlDoc->loadHTML($fetchResult)

может подавлять предупреждения, но как я могу получить эти предупреждения программно?

Ответы [ 4 ]

203 голосов
/ 17 мая 2010

Вызов

libxml_use_internal_errors(true);

до обработки с помощью $xmlDoc->loadHTML()

Это говорит libxml2 не отправлять ошибок и предупреждений через PHP. Затем, чтобы проверить наличие ошибок и обработать их самостоятельно, вы можете обратиться к libxml_get_last_error () и / или libxml_get_errors () , когда будете готовы.

87 голосов
/ 10 июля 2013

Чтобы скрыть предупреждения, вы должны дать специальные инструкции для libxml, который используется для внутреннего анализа:

libxml_use_internal_errors(true);
$dom->loadHTML($html);
libxml_clear_errors();

libxml_use_internal_errors(true) указывает, что вы собираетесь обрабатывать ошибки и предупреждения самостоятельно и не хотите, чтобы они испортили вывод вашего скрипта.

Это не то же самое, что оператор @. Предупреждения собираются за кулисами, и после этого вы можете получить их, используя libxml_get_errors() на тот случай, если вы захотите выполнить регистрацию или вернуть список проблем вызывающей стороне.

Независимо от того, используете ли вы собранные предупреждения или нет, вы всегда должны очищать очередь, вызывая libxml_clear_errors().

Сохранение состояния

Если у вас есть другой код, который использует libxml, возможно, стоит убедиться, что ваш код не изменяет global состояние обработки ошибок; для этого вы можете использовать возвращаемое значение libxml_use_internal_errors() для сохранения предыдущего состояния.

// modify state
$libxml_previous_state = libxml_use_internal_errors(true);
// parse
$dom->loadHTML($html);
// handle errors
libxml_clear_errors();
// restore
libxml_use_internal_errors($libxml_previous_state);
14 голосов
/ 19 июля 2009

Вы можете установить временный обработчик ошибок с помощью set_error_handler

class ErrorTrap {
  protected $callback;
  protected $errors = array();
  function __construct($callback) {
    $this->callback = $callback;
  }
  function call() {
    $result = null;
    set_error_handler(array($this, 'onError'));
    try {
      $result = call_user_func_array($this->callback, func_get_args());
    } catch (Exception $ex) {
      restore_error_handler();        
      throw $ex;
    }
    restore_error_handler();
    return $result;
  }
  function onError($errno, $errstr, $errfile, $errline) {
    $this->errors[] = array($errno, $errstr, $errfile, $errline);
  }
  function ok() {
    return count($this->errors) === 0;
  }
  function errors() {
    return $this->errors;
  }
}

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

// create a DOM document and load the HTML data
$xmlDoc = new DomDocument();
$caller = new ErrorTrap(array($xmlDoc, 'loadHTML'));
// this doesn't dump out any warnings
$caller->call($fetchResult);
if (!$caller->ok()) {
  var_dump($caller->errors());
}
6 голосов
/ 03 августа 2018

Настройка параметров "LIBXML_NOWARNING" и "LIBXML_NOERROR" также работает отлично:

$dom->loadHTML($html, LIBXML_NOWARNING | LIBXML_NOERROR);
...