在线观看不卡亚洲电影_亚洲妓女99综合网_91青青青亚洲娱乐在线观看_日韩无码高清综合久久

鍍金池/ 問(wèn)答/PHP  C  網(wǎng)絡(luò)安全/ php繼承相關(guān)的一個(gè)問(wèn)題

php繼承相關(guān)的一個(gè)問(wèn)題

class A {
    private function foo() {
        echo "a";
    }
    public function test() {
        $this->foo();
    }
}

class B extends A {
    public function foo() {
        echo 'b';
    }
}

$b = new B();
$b->test();

結(jié)果是打印了a,而不是b。

我先闡明我的邏輯,答主們回答前務(wù)必要明白我的意思啊:

  1. A類的private function foo根據(jù)官方文檔的解釋是私有方法不會(huì)被繼承到子類。那么,如果是這樣的話,也就是說(shuō)B類只繼承了A類的test方法,B類現(xiàn)在有兩個(gè)方法,自己的foo方法和從A繼承過(guò)來(lái)的test方法。
  2. 但是,網(wǎng)上還有一種說(shuō)法是,無(wú)論什么訪問(wèn)屬性,都會(huì)被子類繼承,只是private的方法或者屬性子類并訪問(wèn)不到。那么,如果是這樣的話,B類繼承了A類的兩個(gè)方法,并重寫(xiě)了從A類繼承過(guò)來(lái)的foo方法。
  3. 無(wú)論以上兩種可能哪一種是正確的,也就是說(shuō),無(wú)論B是有自己的foo方法還是B重寫(xiě)了foo方法,B的foo的函數(shù)體都是echo 'b';

那么問(wèn)題來(lái)了,為什么執(zhí)行b的test方法,明明是執(zhí)行的$this->foo();,為什么不是B的foo,而是A的foo呢?


已經(jīng)采納回答,采納 @代碼宇宙 的原因請(qǐng)看答案評(píng)論區(qū)域,同時(shí),推薦這片文章:https://www.2cto.com/kf/20100...


3.16 更新
昨天陷入了思維的怪圈,今天再來(lái)總結(jié)一下:

  • 當(dāng)子類繼承父類之后,this的指向是動(dòng)態(tài)綁定的,也就是說(shuō),當(dāng)父類的方法被重寫(xiě)之后,調(diào)用的就是子類的方法,沒(méi)有重寫(xiě),就調(diào)用父類的方法。就是這么簡(jiǎn)單。
  • 在這個(gè)問(wèn)題中,父類的foo方法是私有的,并沒(méi)有繼承給子類,也就沒(méi)有重寫(xiě)之說(shuō),所以,根據(jù)前面的定論,沒(méi)有重寫(xiě)就調(diào)用父類中的方法。所以就打印了'a'。
  • 另外還有三種綁定方式,分別是 self parent static。

    • self靜態(tài)綁定,它的指向始終是在編譯時(shí)所在的那個(gè)類。
    • parnet也是靜態(tài)綁定,它始終指向編譯時(shí)所在類的父類,而這個(gè)父類是聲明類時(shí)就指定了的,所以指向是明確的。
    • static屬于后期靜態(tài)綁定,或者說(shuō)是動(dòng)態(tài)綁定或運(yùn)行時(shí)綁定,它的指向與調(diào)用者有關(guān)。若調(diào)用者是所在類的實(shí)例,那么它就指向本類;若調(diào)用者是所在類子類的實(shí)例,它則指向子類。
回答
編輯回答
尕筱澄
  1. A類中的foo()是private,不能被繼承,所以不存在重寫(xiě);
  2. test()繼承于A類,由于foo()是不能繼承的,所以B中的foo()可以認(rèn)為是一個(gè)全新的方法;
  3. 當(dāng)A中的foo()從private變?yōu)榭衫^承的時(shí)候,B中的foo()就屬于foo()的重寫(xiě)了;
  4. 這樣想調(diào)用A中的foo()的話只能用parent::foo();
結(jié)論: test()是A的,$this也是A的,調(diào)用自己私有的foo()很正常嘛。
延伸:為什么A中的foo()改為public結(jié)果不一樣了呢?
因?yàn)锽是繼承A的,B把foo()繼承又重寫(xiě)了,所以A中的foo()不能再用$this訪問(wèn)了,只能用parent::foo()

不能繼承是關(guān)鍵。

2017年4月8日 18:11
編輯回答
陌離殤

B沒(méi)有從A繼承私有方法!

繼承是什么意思?我繼承了,我就有了所有權(quán),我可以使用并修改,這才是繼承。使用就是調(diào)用,修改就是重寫(xiě),缺任何一個(gè)都不叫繼承。

那A的私有方法與B有什么關(guān)系呢?就是B只有使用權(quán),并且只能通過(guò)A去使用它,只有這么一點(diǎn)關(guān)系。B既無(wú)法直接調(diào)用,也無(wú)法修改。

所以,官網(wǎng)是正確的,嚴(yán)謹(jǐn)?shù)?。網(wǎng)上的說(shuō)法則是不嚴(yán)謹(jǐn)?shù)摹?/p>

2017年12月11日 12:09
編輯回答
心上人

原因是這樣的,你B繼承了Atest這個(gè)方法你繼承下來(lái)了,但是test本身是在A類里面的,也就是說(shuō),你繼承了test,但是test依舊是屬于A的,那么此時(shí)test里的$this指的就是A這個(gè)對(duì)象。

如果你想要在B里調(diào)用B自己的foo的話,你可以這么做:

class B extends A {
    public function foo() {
        echo 'b';
    }
    public function test() {
        parent::test(); // 調(diào)用父類的test方法,父類里的test方法調(diào)用父類自身的foo方法
        $this->foo(); // 掉用自己的foo方法
    }
}

整體來(lái)說(shuō)的話,就是不要把繼承當(dāng)成屬于.

另外就是官方說(shuō)的是正確的,你這里并不是繼承問(wèn)題,而是應(yīng)該弄清楚屬于與不屬于問(wèn)題。

根據(jù)你的回復(fù),就這個(gè)我繼續(xù)做一下回答:


看一下你本身的這個(gè)例子

class A {
    private function foo() {
        echo 'a';
    }
    public function test() {
        $this->foo();
    }
}
class B extends A {
    public function foo() {
        echo 'b';
    }
}

在這種情況下,如果你把A中的foo方法改成了public,那么此時(shí)foo就被繼承了,而在B類里的foo方法相當(dāng)于是重寫(xiě)了A類的foo方法,但是此時(shí)A類中的foo并沒(méi)有消失,但是需要在B類中以parent::foo()這種形式來(lái)訪問(wèn)父類方法。這種情況方便理解,可說(shuō)A中的foo方法被隱藏了。

既然被重寫(xiě)了,那么自然就會(huì)調(diào)用foo就是調(diào)用重寫(xiě)后的foo了,可以理解為既然有新的東西,那么就肯定用新東西了,新的自然優(yōu)先嘛。

然后回頭看為什么A中原本為private,雖然B繼承了一個(gè)公共方法還是訪問(wèn)A類中的foo呢?首先我們看一下前提概要,也就是A類中的foo是private的,B類不可繼承,那么此時(shí)就相當(dāng)于有兩個(gè)foo,一個(gè)是private的,一個(gè)是public的,嗯,這里要區(qū)分private與"隱藏"是不同的概念。

那么此時(shí)其實(shí)就相當(dāng)于是,雖然B繼承了A,但是A會(huì)首先到自己里面找,那么此時(shí)foo沒(méi)有被重寫(xiě),也就是沒(méi)有被覆蓋,那么肯定是調(diào)用自己的了。對(duì)于A來(lái)說(shuō),B你繼承了我,但是我自己有,我就不要你的東西了,還是自己老婆好啊。

嗯。。。我試著用幾句更簡(jiǎn)單的話來(lái)描述看看是否可行:

A: 我是A?
B: 我是B,我繼承你了啊,我繼承了你的test,我有foo
A:嗯,我也有foo,我不給你用
B:我被調(diào)用了啊,用的是繼承了你的test
A:嗯,我看看,哦,調(diào)用了foo啊,我有
B:能用我的foo嗎?
A:不能
B:好吧。。。?

再來(lái)看把A中的private變?yōu)閜ublic后:

A:我是A
B:我是B,我繼承了你的test,也繼承了你的foo,然而我把你的foo重寫(xiě)了
A:我艸你大爺?,那你用parent::foo()調(diào)用我給你的foo了么?
B:。。。沒(méi)有,我現(xiàn)在是大爺?
A:好吧。。。
B:我被調(diào)用了啊,用的是繼承了你的test
A:好吧...臥槽,foo被你重寫(xiě)了啊,那用你的foo
B:那是當(dāng)然啊,要是還要用你的,我會(huì)用parent::foo()調(diào)用的
A:行吧,艸你大爺?

2018年1月25日 08:39
編輯回答
陌上花

因?yàn)閏lass A的私有方法不能被覆蓋,所以,在class B中再次定義時(shí),它的作用域是不同的,而 A.test只能看到A.foo.當(dāng)你把 A.foo的屬性改為 protectedpublic,你就能看到0

2017年12月22日 19:27