ホームページ >バックエンド開発 >PHPチュートリアル >サブクラス名を取得するPHP継承方法_PHPチュートリアル
データストアの概要を見て、MVC メソッドを使用して PHP を書き始めたので、データストアのいくつかの基本的な機能を実現できる Redis のモデルを PHP を使用して作成したいと思いました...そこで、このようなものに遭遇しました。問題-.-
PHP の __CLASS__ のようなものは、サブクラスでオーバーロードされていない場合、親クラスのメソッドを継承することによって取得されるのは、サブクラスの名前ではなく、親クラスの名前になります。例:
クラスA{
関数 __construct(){
エコー __CLASS__;
}
静的関数名(){
エコー __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 = デバッグ_バックトレース();
//次のように、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])) {
$toret = $bt[3][args][0][0];
$toret を返します;
}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][行]) {
自分::$i++;
} その他 {
自分::$i = 0;
self::$fl = $bt[2][ファイル].$bt[2][行];
}
$lines = ファイル($bt[2][ファイル]);
preg_match_all(
/([a-zA-Z0-9_]+)::.$bt[2][関数]./,
$lines[$bt[2][line]-1],
$マッチ
);
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は現在のクラス名を取得するメソッドを直接継承できるようになります~