PHP 유형 힌팅에서 "catchable fatal error"를 어떻게 잡을 수 있습니까?


96

내 클래스 중 하나에서 PHP5의 Type Hinting을 구현하려고합니다.

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

class ClassB {}
class ClassWrong{}

올바른 사용법 :

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

생성 오류 :

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

캐치 가능한 치명적 오류 : ClassA :: method_a ()에 전달 된 인수 1은 ClassB의 인스턴스 여야하며 ClassWrong의 인스턴스가 지정되어야합니다.

그 오류를 잡을 수 있습니까 ( "catchable"이라고 말했기 때문에)? 그렇다면 어떻게?


4
향후 참조 : 엔진의 예외 (PHP 7 용)-PHP 7 부터 치명적인 오류를 포착 할 수 있습니다. 이것은 PHP 7부터 잡히기 때문에 여기서 논의 된 “Catchable fatal error” ( E_RECOVERABLE_ERROR)에 대한 것입니다.
hakre

답변:


113

업데이트 : 이것은 PHP 7에서 더 이상 잡을 수있는 치명적인 오류가 아닙니다. 대신 "예외"가 발생합니다. Exception 에서 파생 된 것이 아니라 Error 에서 파생 된 "예외"(두려운 따옴표로 묶음) . 여전히 Throwable 이며 일반 try-catch 블록으로 처리 할 수 ​​있습니다. 참조 https://wiki.php.net/rfc/throwable-interface를

<?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.';

인쇄물

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.

php7 이전 버전에 대한 이전 답변 :
http://docs.php.net/errorfunc.constants 말한다 :

E_RECOVERABLE_ERROR (integer)
캐치 가능한 치명적인 오류입니다. 위험한 오류가 발생했지만 엔진을 불안정한 상태로 두지 않았 음을 나타냅니다. 사용자 정의 핸들 ( set_error_handler () 참조)에 오류가 발견되지 않으면 E_ERROR이므로 응용 프로그램이 중단됩니다.

참조 : http://derickrethans.nl/erecoverableerror.html

예 :

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.';

인쇄물

'catched' catchable fatal error
done.

편집 : 그러나 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.';

참조 : http://docs.php.net/ErrorException


1
물론 이것은 서버 로그를 볼 때 찾을 수 없다는 점을 제외하면 치명적인 오류와 매우 유사하게 작동합니다. 감사 PHP : /
존 헌트

3
즉, 잡을 수있는 오류를 포착 할 수 없습니다. 훌륭한!
Paul d' Aoust

@Paul이 결론을 내리는 이유는 무엇입니까?
VolkerK

3
오, 나는 단지 그것이 전통적인 의미에서 잡을 수 없다는 것을 의미했습니다 (try / catch 블록 사용). 나는 그날 PHP에 대해 심술 궂은 느낌이 들었 기 때문에 완전히 다른 의미에서 '잡을 수있다'는 것을 알았을 때 논평을 해야겠다고 느꼈다. 당신의 훌륭한 답변에 반대하는 것은 없습니다 (사실 제가 찬성했습니다); 내 모든 분노는 PHP 그 자체였습니다!
Paul d' Aoust

그리고 나는 내가 뭔가 ;-) 간과 한 생각 blog.codinghorror.com/php-sucks-but-it-doesnt-matter D :
VolkerK
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.