PHP では、他の言語でコンパイル エラーを生成するポリモーフィック コードを使用できます。これを簡単に説明します。予想されるコンパイル エラーを生成する最初の C++ コード:
class Base {};
class CommonDerivedBase {
public:
// The "= 0" makes the method and class abstract
// virtual means polymorphic method
virtual whoami() = 0;
};
class DerivedBase : public CommonDerivedBase {
public:
void whoami() { cout << "I am DerivedBase \n"; }
};
class Derived1 : public CommonDerivedBase {
public:
void whoami() { cout << "I am Derived1\n"; }
};
class Derived2 : public CommonDerivedBase {
public:
void whoami() { cout << "I am Derived2\n"; }
};
/* This will not compile */
void test_error(Base& db)
{
db.whoami();
}
C++ コンパイラは、次の行に対してこのエラー メッセージを発行します。db.whoami()
error: no member named 'whoami' in 'Base'
Base には whoami() というメソッドがないためです。ただし、類似の PHP コードは、実行時までそのようなエラーを検出しません。
class Base {}
abstract class DerivedCommonBase {
abstract function whoami();
}
class Derived1 extends DerivedCommonBase {
public function whoami() { echo "I am Derived1\n"; }
}
class Derived2 extends DerivedCommonBase {
public function whoami() { echo "I am Derived2\n"; }
}
/* In PHP, test(Base $b) does not give a runtime error, as long as the object
* passed at run time derives from Base and implements whoami().
*/
function test(Base $b)
{
$b->whoami();
}
$b = new Base();
$d1 = new Derived1();
$d2 = new Derived2();
$a = array();
$a[] = $d1;
$a[] = $d2;
foreach($a as $x) {
echo test($x);
}
test($d1);
test($d2);
test($b); //<-- A run time error will result.
foreach ループは出力で動作します
I am Derived1
I am Derived2
test($b) を呼び出して Base のインスタンスを渡すまで、実行時エラーは発生しません。したがって、foreachの後、出力は次のようになります
I am Derived1
I am Derived2
PHP Fatal error: Call to undefined method Base::whoami() in
home/kurt/public_html/spl/observer/test.php on line 22
PHP をより安全にするためにできる唯一のことは、$b が意図したクラスのインスタンスであるかどうかをテストする実行時チェックを追加することです。
function test(Base $b)
{
if ($b instanceof DerivedCommonBase) {
$b->whoami();
}
}
しかし、ポリモーフィズムの要点は、そのような実行時チェックを排除することです。