亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 網站 > 建站經驗 > 正文

詳細解析Python中__init__()方法的高級應用

2024-04-25 20:34:01
字體:
來源:轉載
供稿:網友

通過工廠函數對 __init__() 加以利用

我們可以通過工廠函數來構建一副完整的撲克牌。這會比枚舉所有52張撲克牌要好得多,在Python中,我們有如下兩種常見的工廠方法:

定義一個函數,該函數會創建所需類的對象。

定義一個類,該類有創建對象的方法。這是一個完整的工廠設計模式,正如設計模式書所描述的那樣。在諸如Java這樣的語言中,工廠類層次結構是必須的,因為該語言不支持獨立的函數。

在Python中,類并不是必須的。只是當有相關的工廠非常復雜的時候才會顯現出優勢。Python的優勢就是當一個簡單的函數可以做的更好的時候我們決不強迫使用類層次結構。

雖然這是一本關于面向對象編程的書,但函數真是一個好東西。這在Python中是常見的也是最地道的。

如果需要的話,我們總是可以將一個函數重寫為適當的可調用對象。我們可以將一個可調用對象重構到我們的工廠類層次結構中。我們將在第五章《使用可調用對象和上下文》中學習可調用對象。

一般,類定義的優點是通過繼承實現代碼重用。工廠類的函數就是包裝一些目標類層次結構和復雜對象的構造。如果我們有一個工廠類,當擴展目標類層次結構的時候,我們可以添加子類到工廠類中。這給我們提供了多態性工廠類;不同的工廠類定義具有相同的方法簽名,可以交替使用。

這類水平的多態性對于靜態編譯語言如Java或C++非常有用。編譯器可以解決類和方法生成代碼的細節。

如果選擇的工廠定義不能重用任何代碼,則在Python中類層次結構不會有任何幫助。我們可以簡單的使用具有相同簽名的函數。

以下是我們各種Card子類的工廠函數:

def card(rank, suit):

if rank == 1:

return AceCard('A', suit)

elif 2 <= rank < 11:

return NumberCard(str(rank), suit)

elif 11 <= rank < 14:

name = {11: 'J', 12: 'Q', 13: 'K' }[rank]

return FaceCard(name, suit)

else:

raise Exception("Rank out of range")

這個函數通過數值類型的rank和suit對象構建Card類。我們現在可以非常簡單的構建牌了。我們已經封裝構造問題到一個單一的工廠函數中,允許應用程序在不知道精確的類層次結構和多態設計是如何工作的情況下進行構建。

下面是一個如何通過這個工廠函數構建一副牌的示例:

deck = [card(rank, suit) for rank in range(1,14) for suit in (Club, Diamond, Heart, Spade)]

它枚舉了所有的牌值和花色來創建完整的52張牌。
1. 錯誤的工廠設計和模糊的else子句

注意card()函數里面的if語句結構。我們沒有使用“包羅萬象”的else子句來做任何處理;我們只是拋出異常。使用“包羅萬象”的else子句會引出一個小小的辯論。

一方面,從屬于else子句的條件不能不言而喻,因為它可能隱藏著微妙的設計錯誤。另一方面,一些else子句確實是顯而易見的。

重要的是要避免含糊的else子句。

考慮下面工廠函數定義的變體:

def card2(rank, suit):

if rank == 1:

return AceCard('A', suit)

elif 2 <= rank < 11:

return NumberCard(str(rank), suit)

else:

name = {11: 'J', 12: 'Q', 13: 'K'}[rank]

return FaceCard(name, suit)

以下是當我們嘗試創建整副牌將會發生的事情:

deck2 = [card2(rank, suit) for rank in range(13) for suit in (Club, Diamond, Heart, Spade)]

它起作用了嗎?如果if條件更復雜了呢?

一些程序員掃視的時候可以理解這個if語句。其他人將難以確定是否所有情況都正確執行了。

對于高級Python編程,我們不應該把它留給讀者去演繹條件是否適用于else子句。對于菜鳥條件應該是顯而易見的,至少也應該是顯示的。

何時使用“包羅萬象”的else

盡量的少使用。使用它只有當條件是顯而易見的時候。當有疑問時,顯式的并拋出異常。

避免含糊的else子句。

2. 簡單一致的使用elif序列

我們的工廠函數card()是兩種常見工廠設計模式的混合物:

if-elif序列

映射

為了簡單起見,最好是專注于這些技術的一個而不是兩個。

我們總是可以用映射來代替elif條件。(是的,總是。但相反是不正確的;改變elif條件為映射將是具有挑戰性的。)

以下是沒有映射的Card工廠:

def card3(rank, suit):

if rank == 1:

return AceCard('A', suit)

elif 2 <= rank < 11:

return NumberCard(str(rank), suit)

elif rank == 11:

return FaceCard('J', suit)

elif rank == 12:

return FaceCard('Q', suit)

elif rank == 13:

return FaceCard('K', suit)

else:

raise Exception("Rank out of range")

我們重寫了card()工廠函數。映射已經轉化為額外的elif子句。這個函數有個優點就是它比之前的版本更加一致。

3. 簡單的使用映射和類對象

在一些示例中,我們可以使用映射來代替一連串的elif條件。很可能發現條件太復雜,這個時候或許只有使用一連串的elif條件來表達才是明智的選擇。對于簡單示例,無論如何,映射可以做的更好且可讀性更強。

因為class是最好的對象,我們可以很容易的映射rank參數到已經構造好的類中。

以下是僅使用映射的Card工廠:

def card4(rank, suit):

class_= {1: AceCard, 11: FaceCard, 12: FaceCard, 13: FaceCard}.get(rank, NumberCard)

return class_(rank, suit)

我們已經映射rank對象到類中。然后,我們傳遞rank值和suit值到類來創建最終的Card實例。

最好我們使用defaultdict類。無論如何,對于微不足道的靜態映射不會比這更簡單了??雌饋硐裣旅娲a片段那樣:

defaultdict(lambda: NumberCard, {1: AceCard, 11: FaceCard, 12: FaceCard, 12: FaceCard})

注意:defaultdict類默認必須是零參數的函數。我們已經使用了lambda創建必要的函數來封裝常量。這個函數,無論如何,都有一些缺陷。對于我們之前版本中缺少1到A和13到K的轉換。當我們試圖增加這些特性時,一定會出現問題的。

我們需要修改映射來提供可以和字符串版本的rank對象一樣的Card子類。對于這兩部分的映射我們還可以做什么?有四種常見解決方案:

可以做兩個并行的映射。我們不建議這樣,但是會強調展示不可取的地方。

可以映射個二元組。這個同樣也會有一些缺點。

可以映射到partial()函數。partial()函數是functools模塊的一個特性。

可以考慮修改我們的類定義,這種映射更容易。可以在下一節將__init__()置入子類定義中看到。

我們來看看每一個具體的例子。

3.1. 兩個并行映射

以下是兩個并行映射解決方案的關鍵代碼:

class_= {1: AceCard, 11: FaceCard, 12: FaceCard, 13: FaceCard}.get(rank, NumberCard)

rank_str= {1:'A', 11:'J', 12:'Q', 13:'K'}.get(rank, str(rank))

return class_(rank_str, suit)

這并不可取的。它涉及到重復映射鍵1、11、12和13序列。重復是糟糕的,因為在軟件更新后并行結構依然保持這種方式。

不要使用并行結構

并行結構必須使用元組或一些其他合適的集合來替代。

3.2. 映射到元組的值

以下是二元組映射的關鍵代碼:

class_, rank_str= {

1: (AceCard,'A'),

11: (FaceCard,'J'),

12: (FaceCard,'Q'),

13: (FaceCard,'K'),

}.get(rank, (NumberCard, str(rank)))

return class_(rank_str, suit)

這是相當不錯的。不需要過多的代碼來分類打牌中的特殊情況。當我們需要改變Card類層次結構來添加額外的Card子類時,我們將看到它如何被修改或被擴展。

將rank值映射到一個類對象的確讓人感覺奇怪,且只有類初始化所需兩個參數中的其中之一。將牌值映射到一個簡單的類或沒有提供一些混亂參數(但不是所有)的函數對象似乎會更合理。

3.3. partial函數解決方案

相比映射到二元組函數和參數之一,我們可以創建一個partial()函數。這是一個已經提供一些(但不是所有)參數的函數。我們將從functools庫中使用partial()函數來創建一個帶有rank參數的partial類。

以下是一個映射rank到partial()函數,可用于對象創建:

from functools import partial

part_class= {

1: partial(AceCard, 'A'),

11: partial(FaceCard, 'J'),

12: partial(FaceCard, 'Q'),

13: partial(FaceCard, 'K'),

}.get(rank, partial(NumberCard, str(rank)))

return part_class(suit)

映射將rank對象與partial()函數聯系在一起,并分配給part_class。這個partial()函數可以被應用到suit對象來創建最終的對象。partial()函數是一種常見的函數式編程技術。它在我們有一個函數來替代對象方法這一特定的情況下使用。

不過總體而言,partial()函數對于大多數面向對象編程并沒有什么幫助。相比創建partial()函數,我們可以簡單地更新類的方法來接受不同組合的參數。partial()函數類似于給對象構造創建一個連貫的接口。

3.4. 連貫的工廠類接口

在某些情況下,我們設計的類為方法的使用定義了順序,衡量方法的順序很像partial()函數。

在一個對象表示法中我們可能會有x.a() .b()。我們可以把它當成x(a, b)。x.a()函數是等待b()的一類partial()函數。我們可以認為它就像x(a)(b)那樣。

這里的想法是,Python給我們提供兩種選擇來管理狀態。我們既可以更新對象又可以創建有狀態性的(在某種程度上)partial()函數。由于這種等價,我們可以重寫partial()函數到一個連貫的工廠對象中。我們使得rank對象的設置為一個連貫的方法來返回self。設置suit對象將真實的創建Card實例。

以下是一個連貫的Card工廠類,有兩個方法函數,必須在特定順序中使用:

class CardFactory:

def rank(self, rank):

self.class_, self.rank_str= {

1: (AceCard, 'A'),

11: (FaceCard,'J'),

12: (FaceCard,'Q'),

13: (FaceCard,'K'),

}.get(rank, (NumberCard, str(rank)))

return self

def suit(self, suit):

return self.class_(self.rank_str, suit)

rank()方法更新構造函數的狀態,suit()方法真實的創建了最終的Card對象。

這個工廠類可以像下面這樣使用:

card8 = CardFactory()

deck8 = [card8.rank(r+1).suit(s) for r in range(13) for s in (Club, Diamond, Heart, Spade)]

首先,我們創建一個工廠實例,然后我們使用那個實例創建Card實例。這并沒有實質性改變__init__()本身在Card類層次結構中如何運作的。然而,它確實改變了我們客戶端應用程序創建對象的方式。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲欧美日韩爽爽影院| 在线看日韩av| 2019亚洲男人天堂| 亚洲天堂av在线免费| 91国内免费在线视频| 这里只有精品丝袜| 亚洲国产天堂久久国产91| 精品中文字幕乱| 日本三级韩国三级久久| 91夜夜未满十八勿入爽爽影院| 91精品久久久久久久久久另类| 国模吧一区二区三区| 欧美一区三区三区高中清蜜桃| 色妞久久福利网| 久久精品成人一区二区三区| 亚洲精品美女久久久久| 91禁外国网站| 国产精品久久久久久久美男| 国产99在线|中文| 97成人精品视频在线观看| 欧美亚洲免费电影| 成人在线观看视频网站| 夜夜嗨av色综合久久久综合网| 久久人人爽人人爽人人片av高请| 欧美视频在线观看免费| 日韩中文字幕在线免费观看| 久久久久久久色| 精品成人在线视频| 一区二区亚洲欧洲国产日韩| 色悠久久久久综合先锋影音下载| 午夜精品一区二区三区在线视| 欧美在线亚洲一区| 日韩成人av在线播放| 亚洲va男人天堂| 欧美孕妇毛茸茸xxxx| 久久天天躁狠狠躁夜夜av| 亚洲精品大尺度| 久久久成人的性感天堂| 国产性猛交xxxx免费看久久| 国产精品一区二区性色av| www.xxxx精品| 中文字幕日韩欧美精品在线观看| 国产日韩在线看片| 欧美性猛交xxxx富婆| 中文亚洲视频在线| 精品国产一区二区三区四区在线观看| 亚洲国产天堂网精品网站| 久久久精品在线| 亚洲美女性生活视频| 久久天天躁狠狠躁夜夜爽蜜月| 欧洲成人在线视频| 国内揄拍国内精品| 在线观看中文字幕亚洲| 国产一区二区三区四区福利| 国产亚洲精品美女久久久久| 欧美激情2020午夜免费观看| 精品久久香蕉国产线看观看gif| 色妞欧美日韩在线| 91老司机精品视频| 国产精品久久久精品| 国产精品国语对白| 国产精品免费一区二区三区都可以| 国产中文欧美精品| 亚洲国产又黄又爽女人高潮的| 亚洲最大的免费| 91在线视频精品| 欧美成人在线影院| 亚洲第五色综合网| 自拍偷拍亚洲在线| 国产91色在线播放| 日韩av快播网址| 亚洲自拍小视频| 久久久久久久一区二区三区| 国产日韩av在线| 成人免费黄色网| 亚洲国产天堂久久综合| www.亚洲人.com| 日韩av资源在线播放| 91在线视频免费| 日韩av电影手机在线| 国产一区视频在线| 欧美精品一区二区三区国产精品| 久久久久久国产免费| 日韩美女在线看| 欧美日韩国产一区二区| 精品国产区一区二区三区在线观看| 欧美一级大片视频| 日韩av中文字幕在线播放| 91精品视频在线播放| 自拍偷拍亚洲精品| 久久久噜噜噜久久| 午夜精品一区二区三区av| 成人中心免费视频| 久久6免费高清热精品| 欧美第一页在线| 国产一区二区三区在线看| 亚洲精品国精品久久99热一| 91麻豆国产精品| 91久久久久久| 欧美性猛交xxxx黑人猛交| 国产精品嫩草视频| 欧美日韩国产综合视频在线观看中文| 亚洲国产91精品在线观看| 91精品久久久久久久久久久久久| 久久久免费观看| 国产精品中文字幕久久久| 欧美一级视频免费在线观看| 成人免费黄色网| 日韩电影在线观看永久视频免费网站| 91禁国产网站| 欧美日韩人人澡狠狠躁视频| 国产精品日韩在线观看| 91在线播放国产| 欧美国产欧美亚洲国产日韩mv天天看完整| 亚州精品天堂中文字幕| 一区二区三区视频观看| 国产日韩在线看片| 午夜精品一区二区三区av| 成人黄色在线免费| 欧美刺激性大交免费视频| 欧美日韩国产综合视频在线观看中文| 欧美另类第一页| 日韩在线中文视频| 国产美女精彩久久| 九九热精品视频国产| 久久五月天色综合| 久久久国产成人精品| 91在线看www| 91国产中文字幕| 国产成人精品久久二区二区| 欧美在线中文字幕| 午夜精品蜜臀一区二区三区免费| 国产精品美女在线| 日韩一区二区久久久| 国产这里只有精品| 亚洲女人被黑人巨大进入| 一道本无吗dⅴd在线播放一区| 国产精品美女久久久免费| 黑丝美女久久久| 欧美成人精品一区二区三区| 亚洲精品国产美女| 国产成人91久久精品| 插插插亚洲综合网| 欧美电影院免费观看| 日本aⅴ大伊香蕉精品视频| 色噜噜狠狠色综合网图区| 一本一本久久a久久精品综合小说| 欧美日韩国产二区| 国产精品专区第二| 久久97精品久久久久久久不卡| 色婷婷综合成人| 羞羞色国产精品| 久久在线精品视频| 亚洲缚视频在线观看| 欧美激情第一页xxx| 欧美国产精品日韩| 国产精品视频专区| 欧美精品videossex88| 久久国产一区二区三区| 欧美激情免费视频| 91极品视频在线| 久久噜噜噜精品国产亚洲综合| 久久综合网hezyo| 欧美激情日韩图片|