ホームページ >バックエンド開発 >PHPチュートリアル >サブクラス名を取得するPHP継承方法の解説_PHPチュートリアル
データストアの入門を見て、MVC メソッドを使用して PHP を書き始めたので、データストアのいくつかの基本的な機能を実現できる Redis のモデルを PHP を使用して作成したいと思いました。そこで、このような問題に遭遇しました。 -.-
PHP の __CLASS__ のようなものは、サブクラスでオーバーロードされていない場合、親クラスのメソッドを継承することによって取得されるのは、サブクラスの名前ではなく、親クラスの名前になります。例:
クラスA{
関数 __construct(){
エコー __CLASS__;
}
静的関数名(){
echo __CLASS__;
}
}
クラス B は A を拡張します{}
このとき、Bをインスタンス化しても、静的メソッドを直接呼び出しても、Aはエコーされます。 Qeephp はこの問題を解決するためにサブクラスのオーバーロードを使用しますが、新しいサブクラスを作成しない場合は、クラス名を呼び出す対応するメソッドをオーバーロードする必要があります...これは oop の PHP の欠陥と考えられているため、Python で試してみました。そのような問題はありませんでした。
ググってみてください。 get_class() と get_called_class() の 2 つの関数を見つけます。 get_class() はインスタンス呼び出しに使用され、パラメータを追加します
($this) はサブクラス継承の呼び出しの問題を解決できますが、get_called_class() は静的メソッドの呼び出しに使用されますが、...これは PHP でのみ利用可能です
5.3 以降です... 5.3 はまだ遠いです... 幸いなことに、この関数は 5.2 より前でも手動で実装できます: http://php.net/manual/en を参照してください。
/function.get-called-class.php 以下では、一部の専門家が 5.3 より前にいくつかの実装メソッドを追加しました。
if(!function_exists(get_called_class)) {
クラス class_tools
{
プライベート静的 $i = 0;
プライベート静的 $fl = null;
パブリック静的関数 get_called_class()
{
$bt = debug_backtrace();
// call_user_func または call_user_func_array 関数を使用してクラスメソッドを呼び出します。プロセスは次のとおりです
if (array_key_exists(3, $bt) &&
array_key_exists(関数, $bt[3]) &&
in_array($bt[3][関数], array(call_user_func,
call_user_func_array))
){
//パラメーターが配列の場合
If (is_array ($ bt [3] [args] [0]) {
}else if(is_string($bt[3][args][0])) {//パラメータが文字列の場合
//それが文字列であり、その文字列に :: 記号が含まれている場合、それは正しいパラメータ型とみなされ、クラス名が計算されて返されます
If(false !== strpos($bt[3][args][0], ::)) {
$toret =explode(::, $bt[3][args][0]);
$toret[0];
を返します
}
}
// A::make() などの通常の方法でクラス メソッドを呼び出します
If(self::$fl == $bt[2][ファイル].$bt[2][行]) {
self::$i ;
} else {
self::$i = 0;
}
$lines = file($bt[2][file]);
Preg_match_all(/([a-zA-Z0-9_] )::.$bt[2][関数]./,$lines[$bt[2][line]-1],$matches);
return $matches[1][self::$i];
}
}
関数 get_called_class()
{
return class_tools::get_called_class();
}
}
これで、この例を次のように変更できるようになりました:
クラスA{
関数 __construct(){
echo get_class($this);
}
静的関数名(){
echo get_called_class();
}
}
クラス B は A を拡張します{}
これにより、Bは現在のクラス名を取得するメソッドを直接継承できるようになります~