ホームページ > バックエンド開発 > PHPチュートリアル > 継承されたクラスの this は親クラスを指しているのでしょうか?

継承されたクラスの this は親クラスを指しているのでしょうか?

WBOY
リリース: 2016-06-23 14:19:02
オリジナル
1280 人が閲覧しました

<?phpclass pa{    private function m(){        echo 'Parent\'s function';    }    public function run(){        $this->m();    }}class child extends pa{    public function m(){        echo 'child\'s function';    }}$obj=new child();$obj->run();
ログイン後にコピー


結果は、親関数であることがわかります。ただし、 protected と run() がサブクラスに統合されているため、 m() メソッドは継承されません。メソッド内の this は、 を指すはずです。サブクラスのインスタンスを作成し、サブクラスの m メソッドを呼び出しますが、その結果、private が public に変更されると、サブクラスのメソッド m が呼び出されます。


ディスカッションに返信してください。解決策)

詳細な説明はありません foundextends 動作メカニズムに関する記事、以下は私の推測です

この問題の原因は、表面上は親クラスが継承されているものの、実際には親クラスが存在するためではないかと思います。つまり、サブクラスを介してオブジェクトをインスタンス化して親クラスのメソッドを呼び出すと、そのオブジェクトは実際に親クラスの空間に入ります。その場合、親クラス空間内の $this のスコープは親クラス空間自体になります。 、スペース内の子クラスにまたがることはできません。

この推測の理由は、PHP 変数の参照原理によるものです。つまり、ある変数の値が別の変数への参照である場合、2 つの変数が同じ変数になるわけではありません。それ自体 まだ 2 つの変数があり、1 つが破壊されても、もう 1 つは依然として存在します。

したがって、これに基づいて、サブクラスのインスタンス化されたオブジェクトは、サブクラスのオブジェクトを表すために親クラスで $this を使用すべきではないと思います。これにより、オブジェクトが「リープフロッグ」することになります

同じ文: privateは不可侵です

上記の返信はとても洞察力に富んでいます!

extends の仕組みを詳しく説明した記事は見つかりませんでした。以下は私の推測です

この問題の原因は、表面上は親クラスが継承されているのではないかと思います。つまり、親クラスのメソッドがサブクラスのインスタンス化されたオブジェクトを通じて呼び出された場合、そのメソッドは実際に親クラスの空間に入り、親クラス空間内の $this のスコープは親になります。クラス空間そのものであり、サブクラスの空間にまたがることはできません。

この推測の理由は、PHP 変数の参照原理によるものです。つまり、ある変数の値が別の変数への参照である場合、2 つの変数が同じ変数になるわけではありません。それ自体 まだ 2 つの変数があり、1 つが破壊されても、もう 1 つは依然として存在します。

したがって、これに基づいて、サブクラスのインスタンス化されたオブジェクトは、サブクラスのオブジェクトを表すために親クラスで $this を使用すべきではないと思います。これにより、オブジェクトが「リープフロッグ」する原因になります

public と private の結果。属性がパブリックになり、これがサブクラスを表す場合は異なります。説明していただけますか?

親クラスがクラス A、サブクラスがクラス B の場合、メモリ構造は次のようになると思います

A = クラスA

B = { (クラス A から継承) クラス B } <-- $this

サブクラス オブジェクトを作成するときは、最初に親クラス オブジェクトを作成する必要があり、その後、親クラスのメモリ ブロックが作成されます。サブクラスのメモリ ブロックの一部として使用されます (サブクラス = 親クラスのメモリ構造 + サブクラス自体)

したがって、親クラスで定義された this によって呼び出されるメソッドは、親クラスのメソッドであるか、サブクラスによってオーバーライドされます。親クラス。

サブクラスのインスタンスには run メソッドが定義されていません。アクセスするのは親クラスの public メソッドの run です。そのため、$this->m() は既に親クラスを指しているはずです。オーバーライドされていない親クラスの private m メソッド

B = { (Inherited from Class A) <-- $this Class B }

例えば、以下の例

<?phpclass pa{    public function a(){        $this->sd();  // $this指向父类,而父类中未定义sd方法,子类的sd方法又是私有的,所以没有输出,子类的sd改成public即可输出。    }}class child extends pa{    public function m(){        $this->a();    }    private function sd(){        echo 'sd';    }}$obj=new child();$obj->m();
ログイン後にコピー


深く理解できていないPHP の内部実装、およびガイダンスについての議論は歓迎です。

同じ文です: プライベートは不可侵です

パブリックとプライベートの異なる結果をどう説明すればいいのかわかりません

プライベートには自分だけがアクセスできます。継承も上書きもされません
パブリックは共有され、自由にアクセス、継承、上書きできます


同じ文です: プライベートは不可侵です

パブリックとプライベートの結果の違いをどう説明すればいいのかわかりません

php の場合、public と private は 2 つの異なる変更であるため

つまり、親クラスが private m の場合、サブクラスの public m は実際にはサブクラスの単なる内部メソッドであり、親クラスとは何の関係もありません。 run を呼び出すときは、まず run で宣言された空間に戻り、プライベート メソッドが見つかった場合は、それを直接実行します。それ以外の場合は、現在のオブジェクトの空間に戻ってパブリック メソッドまたはプロテクト メソッドを探します。がまだ見つからず、extends クラスが存在する場合、クラスのスペースに移動してそれを見つけます。それでも見つからない場合は、エラーが報告されます。


extends の仕組みを詳しく説明した記事は見つかりませんでした。以下は私の推測です

この問題の原因は、表面上は親クラスが継承されているのではないかと思います。つまり、親クラスのメソッドがサブクラスのインスタンス化されたオブジェクトを通じて呼び出された場合、そのメソッドは実際に親クラスの空間に入り、親クラス空間内の $this のスコープは親になります。クラス空間そのものであり、サブクラスの空間にまたがることはできません。

この推測の理由は、PHP 変数の参照原理によるものです。つまり、ある変数の値が別の変数への参照である場合、2 つの変数が同じ変数になるわけではありません。それ自体 まだ 2 つの変数があり、1 つが破壊されても、もう 1 つは依然として存在します。

したがって、これに基づいて、サブクラスのインスタンス化されたオブジェクトは、サブクラスのオブジェクトを表すために親クラスで $this を使用すべきではないと思います。これにより、オブジェクトが「リープフロッグ」する原因になります

public と private の結果。属性が public になったとき、これはサブクラスを表します。説明していただけますか

上記の皆さんの説明によると、private はオーバーライドできませんが、public はオーバーライドできるため、public に変更した後、親クラスはサブクラスによってオーバーライドされます



同じ文です: private は不可侵です

パブリックとプライベートの異なる結果を説明する方法がわかりません

なぜなら、php の場合、public と private は次のとおりです。 2 つの異なる変更

つまり、親クラスが private m の場合、サブクラスの public m は実際にはサブクラスの単なる内部メソッドであり、親クラスとは何の関係もありません。 run を呼び出すときは、まず に戻ります。 run で宣言された空間でプライベート メソッドを見つけます。見つかった場合は、直接実行します。見つからない場合は、現在のオブジェクトの空間に戻って、パブリック メソッドまたはプロテクト メソッドを見つけます。クラススペースに移動して見つけてください。それでも見つからない場合は、エラーが報告されます。

run() メソッドはすでにサブクラスのメソッドであるため、 $child->run() は run() 内の this もサブクラスに現れ、サブクラス オブジェクトを参照します。 ->m() は親クラスの m() ではなくサブクラスの m() を呼び出す必要があります。これが私が理解している方法です。それはサブクラスと親クラスの m() とは何の関係もありません。m() が継承されないのはなぜですか。プライベートとパブリックの状況

プライベート、自分だけがアクセスできます。継承またはオーバーライドされません
パブリック共有、自由にアクセス、継承、およびオーバーライドできます

未回答の質問



extends のメカニズムをより詳細に説明する記事は見つかりませんでした。以下は単なる私の考えです。推測します

それがこの問題の原因であると思います。その理由は、親クラスが表面上継承されているにもかかわらず、実際には親クラスが存在する、つまり、親クラスのメソッドがインスタンス化されたオブジェクトを通じて呼び出された場合である可能性があります。サブクラスの場合、実際には親クラスの空間に入ります。親クラス空間内の $this のスコープは親クラス空間そのものであり、サブクラスの空間にまたがることはできません。

この推測の理由は、PHP 変数の参照原理によるものです。つまり、ある変数の値が別の変数への参照である場合、2 つの変数が同じ変数になるわけではありません。それ自体 まだ 2 つの変数があり、1 つが破壊されても、もう 1 つは依然として存在します。

したがって、これに基づいて、サブクラスのインスタンス化されたオブジェクトは、サブクラスのオブジェクトを表すために親クラスで $this を使用すべきではないと思います。これにより、オブジェクトが「リープフロッグ」する原因になります

public と private の結果。属性が public になったとき、これはサブクラスを表します。説明していただけますか

上記の皆さんの説明によると、private はオーバーライドできませんが、public はオーバーライドできるため、public に変更した後、親クラスはサブクラスによってオーバーライドされます
それは this のポイントと何の関係がありますか?




还是那句话:私有的是不可侵犯的

没明白,怎么解释public 和 private结果不同

因为对于php来说 public 和 private 是2种不同的修饰

也就是说,当父类是private m时,子类的public m其实只是子类的内部方法,而和父类完全没有关系,当调用run时,首先回去run声明的空间寻找private的方法,如果找到,则直接执行,否则回去当前对象的空间寻找public或者protected的方法,如果还没有找到,并且有extends class 则会去class的空间找,仍然没有的话,则会报错。

run()方法已经是子类的方法了,那么$child->run()执行的就是子类自己的方法,run()里面的this也是出现在子类中,指代子类对象,那么$this->m()调用的因该是子类的m()而不是父类的m(),我这么理解的。子类和父类的m()没关系,这点我在帖子中已经说了m()没被继承,我的问题是run()方法里面的this指向,为什么在private和public情况下不同,谢谢

run()啥时候是子类方法了,你run是定义在父类里的啊。它的public表示它可以被子类访问和重写,不代表它就直接属于子类的方法啊。

$child->run()执行时$child实例里是没有run方法的。所以按照8楼所说,因为child是子类,所以在子类中找不到时到父类空间中找,这时父类中的run方法中的$this就是表示父类本身。

你使用$this->m()时,由于$this目前表示父类,而m()又是private的,所以不存在被覆盖的可能,子类里定义的是属于子类自己的,跟父类无关。调用的就是父类的m,而不是子类。

当你把父类的m()的private改成public,父类的m()就存在被重写的可能,所以程序会去继承他的子类中去寻找(也就是你的$child),如果找到则调用子类的,没找到调用父类自己的m()

第一种请看等价于:

<?phpclass child{    private function m(){        echo 'Parent\'s function';    }    public function run(){        $this->m();    }    public function m(){        echo 'child\'s function';    }}$obj=new child();$obj->run();
ログイン後にコピー


这样搞也可以吗?

搞错了,请无视。竟然不能编辑。。

$this 是类实例化后的对象
所以 $this 指向的是对象,而不是类
实例化后,无论是基类还是继承类的方法都在对象中
具体执行的是哪个方法由 php 根据规则决定

class pa{  private function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到 pa pa::m Parent's function
pa::m 方法是私有的,不可被覆盖

class pa{  public function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到 pa child::m child's function
pa::m 方法是公有的,可以被覆盖

class pa{  private function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();    parent::run();  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到
child child::m child's function
pa pa::m Parent's function

不能理解就慢慢理解吧。规矩不会因为你而改变
我就偏要再“答非所问”一回!

还是那句话:私有的是不可侵犯的
好精辟啊





还是那句话:私有的是不可侵犯的

没明白,怎么解释public 和 private结果不同

因为对于php来说 public 和 private 是2种不同的修饰

也就是说,当父类是private m时,子类的public m其实只是子类的内部方法,而和父类完全没有关系,当调用run时,首先回去run声明的空间寻找private的方法,如果找到,则直接执行,否则回去当前对象的空间寻找public或者protected的方法,如果还没有找到,并且有extends class 则会去class的空间找,仍然没有的话,则会报错。

run()方法已经是子类的方法了,那么$child->run()执行的就是子类自己的方法,run()里面的this也是出现在子类中,指代子类对象,那么$this->m()调用的因该是子类的m()而不是父类的m(),我这么理解的。子类和父类的m()没关系,这点我在帖子中已经说了m()没被继承,我的问题是run()方法里面的this指向,为什么在private和public情况下不同,谢谢

run()啥时候是子类方法了,你run是定义在父类里的啊。它的public表示它可以被子类访问和重写,不代表它就直接属于子类的方法啊。

$child->run()执行时$child实例里是没有run方法的。所以按照8楼所说,因为child是子类,所以在子类中找不到时到父类空间中找,这时父类中的run方法中的$this就是表示父类本身。

你使用$this->m()时,由于$this目前表示父类,而m()又是private的,所以不存在被覆盖的可能,子类里定义的是属于子类自己的,跟父类无关。调用的就是父类的m,而不是子类。

当你把父类的m()的private改成public,父类的m()就存在被重写的可能,所以程序会去继承他的子类中去寻找(也就是你的$child),如果找到则调用子类的,没找到调用父类自己的m()

这时父类中的run方法中的$this就是表示父类本身。按这么说,你的理论就是,实力化子类的时候,父类也实例化了?this只能引用对象不能引用类,你的意思是创建子类的时候,顺便把父类实例化了?否则this这个东西就是个矛盾,都不存在父类对象,哪存在this

$this 是实例化后的对象
就像 $obj=new child(); 后的 $obj 一样
如果你是在对象内部访问对象的方法和属性的话,总不能老是要把 $obj 传进去吧?
所以就有了 $this 这个载体

其实所有面向对象的语言中都有这个 this,也的确迷糊了众多的初学者。其实没有什么好不理解的
this 就是对象自己

$this 是类实例化后的对象
所以 $this 指向的是对象,而不是类
实例化后,无论是基类还是继承类的方法都在对象中
具体执行的是哪个方法由 php 根据规则决定

class pa{  private function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到 pa pa::m Parent's function
pa::m 方法是私有的,不可被覆盖

class pa{  public function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到 pa child::m child's function
pa::m 方法是公有的,可以被覆盖

class pa{  private function m(){    echo __METHOD__ . ' Parent\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();  }}class child extends pa{  public function m(){   echo __METHOD__ . ' child\'s function';  }  public function run(){    echo __CLASS__ . ' ';    $this->m();    parent::run();  }}$obj=new child();$obj->run();
ログイン後にコピー
ログイン後にコピー
得到
child child::m child's function
pa pa::m Parent's function

不能理解就慢慢理解吧。规矩不会因为你而改变
我就偏要再“答非所问”一回!


所以,版主的意思是,子类实例化前,父类也必须先实例化,否则存在在父类中的this没有父类对象供引用,这是第一段代码的结论。
但是根据第二段代码,父类run里面的this调用了子类对象的方法,所以结论,this指向子类,和第一段代码矛盾

this到底指向父类还是子类对象
那么,父类方法中的this,到底引用的是父类,还是子类对象?(难道存在父类对象),根据php代码的运行结果,明显矛盾,一会指向父类一会指向子类

$this 是实例化后的对象
就像 $obj=new child(); 后的 $obj 一样
如果你是在对象内部访问对象的方法和属性的话,总不能老是要把 $obj 传进去吧?
所以就有了 $this 这个载体

其实所有面向对象的语言中都有这个 this,也的确迷糊了众多的初学者。其实没有什么好不理解的
this 就是对象自己

你说的这个我明白,我真正不明白的地方是,子类继承父类方法function a()(这个方法a()内用到了this),这个继承过来的this,到底是谁的this?13楼的回答看得出来他明白了我是什么意思,但是解释的有矛盾,而版主还是没理解我的意思,我不是不明白private和this的规则,我市不知道在继承过程中,this传到了子类的过程中,指向发生了改变还是怎么回事,你的前2个代码的运行结果,明显是php运行机制中的矛盾点

没有什么子类对象和分类对象
也不存在“子类实例化前,父类也必须先实例化”
实例化的只是一个类
只不过这个类可能包还有继承过来的方法和属性

再重复一遍:this 就是对象自己

既然这样,那么继承过来的子类应该等同于下面把

class child extends pa{    public function m(){        echo 'child\'s function';    }	public function run(){//只继承了run(),m()私有,没继承        $this->m();    }}
ログイン後にコピー

那么this就是指向这个类(child)的对象,可是他用了一个不存在的m()方法




还是那句话:私有的是不可侵犯的

没明白,怎么解释public 和 private结果不同

因为对于php来说 public 和 private 是2种不同的修饰

也就是说,当父类是private m时,子类的public m其实只是子类的内部方法,而和父类完全没有关系,当调用run时,首先回去run声明的空间寻找private的方法,如果找到,则直接执行,否则回去当前对象的空间寻找public或者protected的方法,如果还没有找到,并且有extends class 则会去class的空间找,仍然没有的话,则会报错。

run()方法已经是子类的方法了,那么$child->run()执行的就是子类自己的方法,run()里面的this也是出现在子类中,指代子类对象,那么$this->m()调用的因该是子类的m()而不是父类的m(),我这么理解的。子类和父类的m()没关系,这点我在帖子中已经说了m()没被继承,我的问题是run()方法里面的this指向,为什么在private和public情况下不同,谢谢


最后回复一帖,不想在这个问题多做纠结了

首先,看下面这个例子

<?phpclass pa{    private function m(){        echo 'Parent\'s function';    }}class child extends pa{    public function run(){        if(is_callable($this, 'm')) {			echo 'private function can callable';		} else {			echo 'private function can not callable';		}    }}$obj=new child();if(method_exists($obj, 'm')) {	echo 'private function can extends';} else {	echo 'private function can not extends';}$obj->run();
ログイン後にコピー


接着再看
<?phpfunction print_method($obj) {	$refObj = new ReflectionObject($obj);	foreach($refObj->getMethods() as $refFun) {		echo "Define class name : ", $refFun->getDeclaringClass()->getName(), "\n";		echo "Modifiers for method ", $refFun->name,":\n";		echo $refFun->getModifiers() . "\n";		echo implode(' ', Reflection::getModifierNames($refFun->getModifiers())) . "\n";	}} class pa{    private function m(){        echo 'Parent\'s function';    }    public function run(){        print_method($this);		// 从上面的输出可以看到,$this = $obj,		// 这里隐含一个输出,为什么会隐藏,这就是继承的关系,我们知道public和protected会被子类同名给覆盖		// 但是private又会被保护,不能覆盖,那么其实就是有2份m,一个private m,另外一个就是子类 public m		$this->m();		// 而这里的m调用时,会找那个m调用呢?		// 很简单,就是优先找当前method所定义class处的private,因为有了private就不用考虑继承覆盖的问题		    }	protected}class child extends pa{    public function m(){        echo 'child\'s function';    }}$obj=new child();print_method($obj);$obj->run();
ログイン後にコピー


但愿你能明白。

貌似要你那么做需要延迟静态绑定

私有的无论是不是父类都不能调用

$this 是实例化后的对象
就像 $obj=new child(); 后的 $obj 一样
如果你是在对象内部访问对象的方法和属性的话,总不能老是要把 $obj 传进去吧?
所以就有了 $this 这个载体

其实所有面向对象的语言中都有这个 this,也的确迷糊了众多的初学者。其实没有什么好不理解的
this 就是对象自己
应该是他本身吧!无论这个他是父类或子类、或者别的类?

?爆了~~~ 留名学习中~

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート