成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Laravel深入學習3 - 接口約定

鄒立鵬 / 1020人閱讀

摘要:難道,就沒有強類型特征答案是明確的,肯定有其實是一種強類型和弱類型的結(jié)合體。強類型語言中,編譯器通常提供編譯檢查錯誤的功能,它也是非常有用的。與此同時,強類型的代碼看起來也是很生硬的。

聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味的翻譯,能保證90%的原汁性,另外因為是理解翻譯,肯定會有錯誤的地方,歡迎指正。

歡迎轉(zhuǎn)載,轉(zhuǎn)載請注明出處,謝謝!

接口約定 強類型和弱類型

Water Fowl 這里我不知道該怎么翻譯,按理解就將他翻譯成弱類型?

之前的文章中,涵蓋了依賴注入的基礎(chǔ)知識:什么是依賴注入;如何實現(xiàn);以及他有什么好處。在舉例的代碼中,也展示了如何將接口注入到類之中。在我們繼續(xù)深入之前,有必要深入一下接口相關(guān)的內(nèi)容,因為很多PHP開發(fā)人員對接口都有相當程度的不熟練。

在我成為一個PHP程序員之前,我是寫.NET的。難道我喜歡痛苦或者其他什么的么?在.NET中接口無處不在,事實上.NET框架的很多核心內(nèi)容就是接口,還有一個好處:很多象.NET的語言如C#、VB等都是_強類型_的。通常,傳入方法的原生對象,必須預先定義好相關(guān)的_類型_。例如下面的C#代碼:

public int BillUser(User user)
{
    this.biller.bill(user.GetId(), this.amount)
}

我們不僅定義了傳入?yún)?shù)的_類型_,函數(shù)返回的類型都是已經(jīng)定義好的。C#提倡_安全類型_。函數(shù)BillUser方法傳入的參數(shù)必須是User對象。

PHP是一種_弱類型_語言。在弱類型語言中,對象中可用的方法取決于其方法的使用形式,而非方法繼承或者實現(xiàn)的位置。例如:

public function billUser($user)
{
    $this->biller->bill($user->getId(), $this->amount);
}

我們無需告訴方法類型的參數(shù)是什么,我們可以傳入任意對象,只要他有getId方法就行。這就是弱類型編碼的例子。如果一個東西看起來象鴨子,叫聲也象,那么他就是一個鴨子。換言之,如果一個對象象user,行為也象user,那么他就是一個user對象。

難道,PHP就沒有強類型特征?答案是明確的,肯定有!PHP其實是一種強類型和弱類型的結(jié)合體。為了說明這一點,我們修改下上例的代碼:

public function billUser(User $user)
{
    $this->biller->bill($user->getId(), $amount);
}

在方法參數(shù)中,加入User約定后,就能確保傳入方法的參數(shù)必須是User的實例化對象或者是繼承自User的一個實例。

兩種類型各有優(yōu)劣。強類型語言中,編譯器通常提供編譯檢查錯誤的功能,它也是非常有用的。方法的輸入和輸出都是明確的。

與此同時,強類型的代碼看起來也是很生硬的。比如Eluquent ORM中提供的動態(tài)方法whereEmailOrName就不能象C#那樣明確參數(shù)和返回值的類型。這里不討論孰優(yōu)孰劣,我們各取所長,但是,不假思索的死認某一種方式肯定會埋下很多坑。

示例

接口就是約定,他不包含具體的代碼實現(xiàn),而定義了對象需要實現(xiàn)的一系列的方法。如果一個對象實現(xiàn)了某個接口,那么這個接口的方法肯定都能在這個實例對象中使用。通過約定固化了某些方法的實現(xiàn),這種_多態(tài)_就能保證語言的類型安全。

什么是多態(tài)?

多態(tài)含義很廣,可以理解為一種實體的多種形式。在本書中,我們指代接口的多種實現(xiàn)方式。例如:UserRepositoryInterface可以有MySQL和Redis兩種存儲實現(xiàn)方式,但是每一種都是UserRepositoryInter接口的實現(xiàn)。

為了說明強類型在接口中的靈活和重要性,我們來實現(xiàn)如下一個酒店預訂的例子:

interface ProviderInterface
{
    public function getLowestPrice($location);
    public function book($location);
}

當用戶預訂房間是,我們想將此事件記錄到系統(tǒng)中。我們在User類中添加如下方法:

class User
{
    public function bookLocation(ProviderInterface $provider, $location)
    {
        $amountCharged = $provider->book($location);
        $this->logBookedLocation($location, $amountCharged);
    }
}

我們限定了參數(shù)$provider的類型,User類中就能假定book方法是可安全調(diào)用的,這就使得bookLocation有較強的操作性,我們不用關(guān)心酒店是如何實現(xiàn)房間預訂這一過程。下面的代碼就能體現(xiàn)這一特性:

$location = "Hilton, Dallas";

$cheapestProvider = $this->findCheapest($location, array(
    new PricelineProvider,
    new OrbitzProvider,
));

$user->bookLocation($cheapestProvider, $location);

贊!我們不用關(guān)心那家酒店最便宜,只需要將他傳入User實例中就能成功預訂房間。因為User對象要求傳入的參數(shù)是繼承自ProviderInterface的對象,未來添加更多的酒店提供商,都能使我們的代碼穩(wěn)定的運行。

忘掉細節(jié)

記住,接口_不實現(xiàn)_任何細節(jié),只是簡單的定義類必須實現(xiàn)的方法。

接口和團隊開發(fā)

當團隊構(gòu)建大型應(yīng)用時,不同的模塊進程是不同的。比如,有人處理數(shù)據(jù)層,有人處理前端web、控制器層。前端開發(fā)項測試自己的控制器,但是后端人員開發(fā)進度緩慢。但是,如果我們能約定好接口,后端人員只須遵循接口定義:

interface OrderRepositoryInterface 
{
    public function getMostRecent(User $user);
}

一旦約定了接口,前端開發(fā)人員,在代碼沒有實現(xiàn)的情況下,也能測試自己的控制器!這樣整個應(yīng)用中就不用擔心不同模塊的開發(fā)進度,也不會影響到正常的測試用例的編寫。更深一點來說,這種方法不會影響到其他組件的開發(fā),做到了無知是福。我們不需要讓我們的類必須知道其他類是_怎么_實現(xiàn)的,只需要知道他_能夠_干什么?,F(xiàn)在,我們已經(jīng)定義了接口,那么我們可以繼續(xù)我們控制器代碼的實現(xiàn)了:

class OrderController 
{
    public function __construct(OrderRepositoryInterface $orders)
    {
        $this->orders = $orders;
    }
    
    public function getRecent()
    {
        $recent = $this->orders->getMostRecent(Auth::user());
        return View::make("orders.recent", compact("recent"));
    }
}

前端開發(fā)人員可以自己實現(xiàn)一個“假”接口,來測試應(yīng)用試圖中需要填充的數(shù)據(jù)。

class DummyOrderRepository implements OrderRepositoryInterface 
{
    public function getMostRecent(User $user)
    {
        return array("Order 1", "Order 2", "Order 3");
    }
}

接口實現(xiàn)之后,我們就能將其綁定到容器中,就能在整個應(yīng)用中使用他了:

App::bind("OrderRepositoryInterface", "DummyOrderRepository");

當后端開發(fā)人員實現(xiàn)了他的模塊,比如:RedisOrderRepository。我們再次通過修改綁定將其應(yīng)用到項目之中。

接口大綱

接口在被用來定義項目“骨架”上是非常有用的。在項目組件設(shè)計階段可以促進團隊間設(shè)計討論。比如定義BillingNotifierInterface接口,并討論接口相應(yīng)的方法,在敲代碼前就能用接口定義出一套好的API。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/22952.html

相關(guān)文章

  • Laravel深入學習1 - 依賴注入

    摘要:然而,我們需要注意的是僅是軟件設(shè)計模式依賴注入的一種便利的實現(xiàn)形式。容器本身不是依賴注入的必要條件,在框架他只是讓其變得更加簡便。首先,讓我們探索下為什么依賴注入是有益的。繼續(xù)深入讓我們通過另一個示例來加深對依賴注入的理解。 聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味的翻譯,能保證9...

    sunsmell 評論0 收藏0
  • Laravel深入學習11 - 接口分離原則

    摘要:實際上,本原則要求接口必須是粒度明確的。當你的代碼不符合接口分離原則時,那也肯定違背了單一責任原則。接口分離原則本原則是指在實現(xiàn)類中對于接口中的方法并不強制去實現(xiàn)使用不到的方法。 聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味的翻譯,能保證90%的原汁性,另外因為是理解翻譯,肯定會有錯誤...

    lwx12525 評論0 收藏0
  • Laravel深入學習2 - 控制反轉(zhuǎn)容器

    摘要:控制反轉(zhuǎn)容器控制反轉(zhuǎn)使依賴注入變得更加便捷。有瑕疵控制反轉(zhuǎn)容器是實現(xiàn)的控制翻轉(zhuǎn)容器的一種替代方案。容器的獨立使用即使沒有使用框架,我們?nèi)匀豢梢栽陧椖恐惺褂冒惭b組件來使用的控制反轉(zhuǎn)容器。在沒有給定任何信息的情況下,容器是無法實例化相關(guān)依賴的。 聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味...

    worldligang 評論0 收藏0
  • Laravel深入學習7 - 框架的擴展

    摘要:組件擴展通常有兩種方法向容器中綁定自己的接口實現(xiàn)痛過使用工廠模式實現(xiàn)的類注冊自己的擴展。類庫管理類以工廠模式實現(xiàn),負責諸如緩存等驅(qū)動的實例化。閉包須要傳入繼承自和容器的實例化對象。當完成擴展之后要記住中替換成自己的擴展名稱。 聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味的翻譯,能保證9...

    yuanxin 評論0 收藏0
  • Laravel深入學習12 - 依賴倒置原則

    摘要:在改變存儲系統(tǒng)的情況下,必須對進行修改,違背了開放封閉原則。傳統(tǒng)的依賴痛過倒置就能事代碼變得非常靈活,易于改變 聲明:本文并非博主原創(chuàng),而是來自對《Laravel 4 From Apprentice to Artisan》閱讀的翻譯和理解,當然也不是原汁原味的翻譯,能保證90%的原汁性,另外因為是理解翻譯,肯定會有錯誤的地方,歡迎指正。 歡迎轉(zhuǎn)載,轉(zhuǎn)載請注明出處,謝謝! 依賴反轉(zhuǎn)原則 ...

    IamDLY 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<