怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題

這篇文章主要介紹“怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題”,在日常操作中,相信很多人在怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!

創(chuàng)新互聯(lián)公司專注于成都網(wǎng)站設(shè)計、做網(wǎng)站、網(wǎng)頁設(shè)計、網(wǎng)站制作、網(wǎng)站開發(fā)。公司秉持“客戶至上,用心服務(wù)”的宗旨,從客戶的利益和觀點出發(fā),讓客戶在網(wǎng)絡(luò)營銷中找到自己的駐足之地。尊重和關(guān)懷每一位客戶,用嚴謹?shù)膽B(tài)度對待客戶,用專業(yè)的服務(wù)創(chuàng)造價值,成為客戶值得信賴的朋友,為客戶解除后顧之憂。

細說 PHP 7.2 子類覆蓋方法省略參數(shù)類型功能以及 Liskov 替換原則

PHP 7.2 出來也有段時間了,關(guān)于新版本有什么新改進,只要你關(guān)心 PHP 的發(fā)展,應(yīng)該都看過。這里只細說一個可能會有誤解的新功能。

PHP 7.2 可以在當子類覆蓋(override)父類方法的時候,忽略父類方法的定義的參數(shù)的類型(type hint):

class Foo
{
    public function bar(SomeClass $obj) {}
}
class Foobar extends Foo
{
    public function bar($obj) {} // 這在 PHP7.2 版本之前是會報錯的
}

我看有些網(wǎng)站介紹此功能的時候,說其目的是為了『方便重構(gòu)。如果以后父類方法的參數(shù)類型變了,子類不用再全部換一遍』。聽起來好像很有道理。按這說法,隱含的意思是:如果子類忽略了父類方法參數(shù)類型,被調(diào)用時還是會檢查參數(shù)類型。實際情況是不是這樣做一下實驗就知道了:

<?php
class Foo
{
}
class Bar
{
    public function setFoo(Foo $foo)
    {
    }
}
class BarKid extends Bar
{
    public function setFoo($foo)
    {
    }
}
$kid = new BarKid;
$kid->setFoo('I am a string!');

如果上面的說法是對的,setFoo 接受字符串參數(shù)的時候就應(yīng)該報錯,然而上面代碼在 7.2 下并沒有任何報錯信息,但如果子類的 setFoo 方法加上了參數(shù)類型,就會立馬報錯了。記住網(wǎng)上很多說法都不可信,除了我這個小站……

上面的實驗說明子類方法可省略參數(shù)類型,其目的肯定不是為了方便重構(gòu)。那真正目的是什么呢?

在 PHP 7.1 里有一個新功能,是『可設(shè)置方法或函數(shù)的參數(shù)和返回類型是否可以為 null』。其中有一條看上去比較別扭的規(guī)則:『子類方法參數(shù)類型范圍放寬(即父類參數(shù)若不能為 null ,子類參數(shù)可支持 null),但返回類型縮緊(父類若不能返回 null,子類必須也不行;若父類可以返回 null,子類可以不返回 null)』,當時我很簡單說了一句,是因為 『Liskov 替換原則』,但沒有做深入介紹。身邊的 PHPer 們關(guān)注 OOP 原則的不多,但我認為它應(yīng)該被每個工程師知道,還是介紹一下。

Liskov 替換原則簡單一句話:父類出現(xiàn)的地方,替換成子類也能運行,即子類可無腦替換父類。其實從語言設(shè)計來說,我認為此原則就是對自然規(guī)則的模仿2018-09-29 補充:也不是簡單的『模仿』,有興趣可閱讀新博客『企鵝不是鳥』。

舉個例子,人可以喝酒,喝茶,喝可樂,喝各種飲料,但人作為哺乳動物,怎么著都能喝水吧?但反過來,哺乳動物能喝水,但不一定能喝酒喝茶喝可樂,所以人是哺乳動物的子類。

從語言設(shè)計的角度來說,子類就應(yīng)該是父類的加強版,就是要能比父類處理更多的對象類型,而被覆寫的方法參數(shù)類型的擴大,也是這一原則的體現(xiàn)。

再來說可能有點繞的返回類型,為什么子類要縮小返回的范圍呢?其實只要假設(shè)一個方法的返回會作為另外一個方法的參數(shù),就很好想了。比如一個『水果飲料廠』類,有一個『生產(chǎn)』方法,返回『水果汁』,并傳給了『小朋友』的『喝』方法。有一個『橘子汁工廠』類屬于『水果飲料廠』的子類,它的『生產(chǎn)』方法返回類型縮緊,只能返回『橘子汁』,依然給『小朋友』『喝』,并不會出現(xiàn)任何問題。

再舉一個反例。如果又出現(xiàn)一個『水果飲料廠』的子類,其『生產(chǎn)』方法除了返回水果汁,還能返回果釀酒,那這個子類很顯然不能冒著給小朋友喝酒的風(fēng)險去替換父類。

說完了 Liskov 替換原則,我們再來看看 7.2 里的這個改進,我們這時應(yīng)該知道其實這也是 Liskov 原則的體現(xiàn)。目前來說,替換原則在 PHP 的實現(xiàn)并不完全。可能有人覺得這個版本是不是也支持『父類沒有返回類型,子類可以有返回類型』呢?遺憾的是至少在 7.2 這個版本,并不支持,大家可以自行實驗一下。

7.2 的另外一個新功能,是 object 可以作為任何對象的類型。見官方提供例子:

<?php
function test(object $obj) : object
{
    return new SplQueue();
}
test(new StdClass());

其實在 7.2 發(fā)布之前,也是出于替換原則,有過一次關(guān)于『是否子類可以用 object 類型來替代被覆蓋的方法對象參數(shù)的類型』,但最終投票并沒有通過。雖然我不知道原因,但起碼有人提了。

另外目前 PHP 不能像 Java 那樣重載(overload),沒有辦法可以指定覆蓋的方法的類型(目前只能把類型直接去掉,有點太粗暴):

<?php
class Foo
{
}
class FooFoo extends Foo
{
}
class Bar
{
    public function foo(FooFoo $foo)
    {
    }
}
class BarBar extends Bar
{
    public function foo(Foo $foo) // 依然會報『子類不兼容父類方法格式』的錯誤
    {
    }
}

到此,關(guān)于“怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注創(chuàng)新互聯(lián)網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

網(wǎng)站標題:怎么理解PHP7.2忽略父類方法以及Liskov替換原則相關(guān)問題
分享URL:http://muchs.cn/article28/gespcp.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站制作、小程序開發(fā)網(wǎng)站策劃、外貿(mào)網(wǎng)站建設(shè)App開發(fā)、響應(yīng)式網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都seo排名網(wǎng)站優(yōu)化