Como posso detectar um “erro fatal capturável” nas dicas de tipo PHP?

96

Estou tentando implementar dicas de tipo de PHP5 em uma das minhas aulas,

class ClassA {
    public function method_a (ClassB $b)
    {}
}

class ClassB {}
class ClassWrong{}

Modo de usar correto:

$a = new ClassA;
$a->method_a(new ClassB);

produzindo erro:

$a = new ClassA;
$a->method_a(new ClassWrong);

Erro fatal capturável: o argumento 1 passado para ClassA :: method_a () deve ser uma instância de ClassB, instância de ClassWrong fornecida ...

É possível detectar esse erro (uma vez que diz "capturável")? e se sim como?

hoball
fonte
4
Para referência futura: Exceções no motor (para PHP 7) - A partir do PHP 7 é possível pegar erros fatais. Isso também é para o aqui discutido "Erro fatal capturável" ( E_RECOVERABLE_ERROR), pois eles podem ser detectados a partir do PHP 7 ..
hakre

Respostas:

113

Atualização: Este não é mais um erro fatal capturável no php 7. Em vez disso, uma "exceção" é lançada. Uma "exceção" (entre aspas) que não é derivada de Exceção, mas de Erro ; ainda é um Throwable e pode ser tratado com um bloco try-catch normal. consulte https://wiki.php.net/rfc/throwable-interface

Por exemplo

<?php
class ClassA {
  public function method_a (ClassB $b) { echo 'method_a: ', get_class($b), PHP_EOL; }
}
class ClassWrong{}
class ClassB{}
class ClassC extends ClassB {}


foreach( array('ClassA', 'ClassWrong', 'ClassB', 'ClassC') as $cn ) {
    try{
      $a = new ClassA;
      $a->method_a(new $cn);
    }
    catch(Error $err) {
      echo "catched: ", $err->getMessage(), PHP_EOL;
    }
}
echo 'done.';

estampas

catched: Argument 1 passed to ClassA::method_a() must be an instance of ClassB, instance of ClassA given, called in [...]
catched: Argument 1 passed to ClassA::method_a() must be an instance of ClassB, instance of ClassWrong given, called in [...]
method_a: ClassB
method_a: ClassC
done.

A resposta antiga para versões pré-php7:
http://docs.php.net/errorfunc.constants diz:

E_RECOVERABLE_ERROR (integer)
Erro fatal capturável. Indica que ocorreu um erro provavelmente perigoso, mas não deixou o motor em um estado instável. Se o erro não for detectado por um identificador definido pelo usuário (veja também set_error_handler () ), o aplicativo aborta como se fosse um E_ERROR.

veja também: http://derickrethans.nl/erecoverableerror.html

por exemplo

function myErrorHandler($errno, $errstr, $errfile, $errline) {
  if ( E_RECOVERABLE_ERROR===$errno ) {
    echo "'catched' catchable fatal error\n";
    return true;
  }
  return false;
}
set_error_handler('myErrorHandler');

class ClassA {
  public function method_a (ClassB $b) {}
}

class ClassWrong{}

$a = new ClassA;
$a->method_a(new ClassWrong);
echo 'done.';

estampas

'catched' catchable fatal error
done.

editar: Mas você pode "tornar" uma exceção que você pode tratar com um bloco try-catch

function myErrorHandler($errno, $errstr, $errfile, $errline) {
  if ( E_RECOVERABLE_ERROR===$errno ) {
    echo "'catched' catchable fatal error\n";
    throw new ErrorException($errstr, $errno, 0, $errfile, $errline);
    // return true;
  }
  return false;
}
set_error_handler('myErrorHandler');

class ClassA {
  public function method_a (ClassB $b) {}
}

class ClassWrong{}

try{
  $a = new ClassA;
  $a->method_a(new ClassWrong);
}
catch(Exception $ex) {
  echo "catched\n";
}
echo 'done.';

consulte: http://docs.php.net/ErrorException

VolkerK
fonte
1
Portanto, é claro que isso se comporta de maneira muito semelhante a um erro fatal, exceto quando você olha os logs do servidor, não o encontra. Obrigado php: /
John Hunt
3
então, em outras palavras, você não pode detectar um erro capturável. Maravilhoso!
Paul d'Aoust de
@Paul o que o leva a essa conclusão?
VolkerK
3
Oh, eu só quis dizer que não era capturável no sentido tradicional (usando um bloco try / catch). Eu estava me sentindo mal-humorado por causa do PHP naquele dia, então, quando descobri que ele era 'capturável' em um sentido totalmente diferente, me senti compelido a comentar. Nada contra sua resposta maravilhosa (que na verdade eu votei positivamente); toda a minha ira era pelo próprio PHP!
Paul d'Aoust
E eu pensei que havia esquecido algo ;-) blog.codinghorror.com/php-sucks-but-it-doesnt-matter : D
VolkerK