Mastering Object——oriented Python(python 面向对象编程指南) 笔记

第一部分 用特殊方法实现Python风格的类

为了实现更好的可扩展性,Python语言提供了大量的特殊方法,它们大致分为以下几类。

  • 特性访问
  • 可调用对象
  • 集合
  • 数字
  • 上下文
  • 迭代器

第一章 使用__init()__方法

Python中一切事物皆对象!!!!!!
__init__()方法记住两点:

  • __init()__(初始化)是对象生命周期的开始,每个对象必须正确初始化才能够正常的工作。
  • __init__()可以赋值
    对象的生命周期主要是有创建、初始化、销毁。
    ‘显示而非隐式’:对于每个__init__()方法,都应当显示的制定要初始化的变量。
    每当创建一个对象,python会县床架一个空对象,然后调用该对象的init()函数,提供了初始化的操作。
# 以21点为例作为说明。
class Card(object):
    def __init__(self, suit, rank):
        self.suit = suit
        self.rank = rank
        self.hard, self.soft = self._points()


class NumberCard(Card):
    def _points(self):
        return int(self.rank), int(self.rank)


class AceCard(Card):
    def _points(self):
        return 1, 11


class FaceCard(Card):
    def _points(self):
        return 10, 10
   
class Suit(object):
    def __init__(self,name,symbol):
        self.name = name
        self.symbol = symbol

Club,Diamond,Heart,Spade = Suit('Club','♣'),Suit('Diamond','♦'),Suit('Heart','♥'),Suit('Spade','♠')

通过工厂函数来调用__init__():


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)]
print(deck[0].rank,deck[0].suit.symbol)

这段代码完成了52张牌对象的创建.

使用映射和类来简化设计.

由于类是第一级别的对象,从rank参数射到对象是很容易的事情.
下面的Card类工厂就是使用映射实现的版本.

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

需要修改映射逻辑,除了提供Card子类,还需要提供rank对象的字符串结果.如何实现这两部分映射,有四种常见方案.

  • 可以建立两个并行映射

  • 可以映射为一个二元组.

  • 可以映射为partial()函数.

  • 可以考虑修改类定义的完成映射逻辑.

    1.并行映射

def card5(rank,suit):
    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的逻辑重复.

不要使用并行结构,并行结构应该被元祖或者一些更好的组合所代替

  1. 映射到一个牌面值的元组
def card6(rank,suit):
   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)

从rank值映射到类对象时很少见的,而且两个参数只有一个用于对象的初始化.从rank映射到一个相对简单的类或者是函数对象,而不必提供目的不明确的参数,这才是明智的选择.

3.partial 函数设计

def card7(rank,suit):
  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)

通过调用partial()函数然后复制给part_class,完成于rank对象的管的关联,可以使用同样的方式来创建suit对象,并且完成最终的Card对象的创建.partial()函数的使用在函数时编程中是很常见的.当时用的是函数而非对象方法的时候就可以考虑使用.

大致上,partial()函数在面向对象编程中不是很常用,我们可以简单的的提供构造函数不同版本来做相同的事情.partial()函数和构造对象时的流畅接口很类似.

  1. 工厂模式的流畅的API设计

    有时候我们定义类中的方法必须按照特定的顺序来调用.这种顺序调用的方法和创建 partial() 函数的方式非常类似.

我们可以在流畅接口函数中设置可以返回self值的rank对象,然后传入花色类从而创建Card实例/

以下是Card工厂流畅接口的定义,包含两个函数,他们必须按照顺序调用.

class CardFactory(object):
    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)))
    
    def suit(self,suit):
        return self.class_(self.rank_str,suit)

先使用rank()函数更新了构造函数的状态,然后通过suit()函数创造了 最终的Card对象.

def A (rank):

    a,b ={                      # 本身为一个字典的传递值.返回对应的值.是dict的get方法
        1: (AceCard, 'A'),
        11: (FaceCard, 'J'),
        12: (FaceCard, 'Q'),
        13: (FaceCard, 'K')
    }.get(rank, (NumberCard, str(rank)))
    return a,b # 返回的是一个tuple(),a 为 <class '__main__.NumberCard'> , b 为'3'

a = A(3)
print(a)

我们先实例化一个工厂对象,然然后在创建Card实例,这用方式没有利用__init__() 在Card类层级结构的作用,改变的是调用者创建创建对象的方式.

在每个子类中实现__init__()方法

以下代码演示了如何把__init__()方法提到基类Card中实现的过程.然后在子类中可以重用基类的实现.


class Card(object):
    def __init__(self, rank, suit, hard, soft):
        self.rank = rank
        self.suit = suit
        self.hard = hard
        self.soft = soft


class NumberCard(Card):
    def __init__(self, rank, suit):
        super().__init__(str(rank), suit, rank, rank)


class AceCard(Card):
    def __init__(self, rank, suit):
        super(AceCard, self).__init__("A", suit, 1, 11)


class FaceCard(Card):
    def __init__(self, rank, suit):
        super(FaceCard, self).__init__({11: 'J',
                                        12: 'Q',
                                        13: 'K'}[rank], suit, 10, 10)
        

def card10(rank,suit):
    if rank == 1:
        return AceCard(rank,suit)
    elif 2<= rank < 11:
        return NumberCard(rank,suit)
    elif 11<= rank <14:
        return FaceCard(rank,suit)
    
    else:
        raise  Exception('Rank out of range')

在这里重构了基类中的__init__,虽然将它复杂化,但是这样的权衡是正常的.

使用工厂函数封装的复杂性
__init__()方法和工厂函数之间存在一些权衡,通常直接调动比'程序员友好'的__init__()函数并把复杂性分发给工厂函数更好.当需要封装复杂的构造函数逻辑时,考虑使用工厂函数则更好.

简单的组合对象

一个组合对象也可以称作容器.

如果业务逻辑相对简单,为什么定义新类?

类的定义的一个优势是:

  • 类给对象提供了简单的,不需要实现的接口.

设计集合类,通常是下面三种策略:

  • 封装:这个实际是基于现有集合类来定义一个新类,属于外观模式的一个使用场景.
  • 扩展:这个设计是对现有集合类进行扩展,通常使用定义子类的方式来实现.
  • 创建:即重新设计.
    以上是面向对象设计的核心.
封装集合类

以下是对内部集合进行封装设计.

import random
class Deck(object):
   def __init__(self):
       self._cards = [card6(r+1,s) for r in range(13) for s in (Club,Diamond,Heart,Spade)]
       random.shuffle(self._cards)
   def pop(self):
       return self._cards.pop()

d = Deck()

hand = [d.pop(),d.pop()]

一般来说买外观模式或者封装类中的方法实现只是对底层对象相应函数的代理调用.

class Desk3(list):
    def __init__(self, decks=1):
       super(Desk3, self).__init__()
       for i in range(decks):
           self.extend(card6(r + 1, s) for r in range(13) for s in (Club, Diamond, Heart, Spade))
           random.shuffle(self)
           burn = random.random(1,52)
           for i in range(burn):
               self.pop()

这里我们使用了基类的 __init__()函数来创建了一个空集合,然后调用了 self.extrend()来吧多副牌加载到发牌机中.

复杂的组合对象

模拟打牌策略

class Hand:
    def __init__(self,dealer_card):
        self.dealer_card = dealer_card
        self.cards = []
        
    def hard_total(self):
        return sum(c.hard for c in self.cards)
    
    def soft_total(self):
        return sum(c.soft for c in self.cards)
d = Deck()
h = Hand(d.pop())
h.cards.append(d.pop())
h.cards.append(d.pop())

需要一个一个的添加非常不方便

完成组合对象的初始化

__init__()初始化方法应当返回一个完成的对象,当然这个是理想的情况.而这样也带来复杂性,因为要创建的对象内部可能包含了集合,集合里面又包含了其他对象.

通常考虑使用一个流畅的接口来完成逐个讲对象添加到集合的操作,同时将集合对象作为构造函数来完成初始化.例如:

class Hand2:
    def __init__(self, dealer_card, *cards):
        self.dealer_card = dealer_card
        self.cards = list(cards)

    def hard_total(self):
        return sum(c.hard for c in self.cards)

    def soft_total(self):
        return sum(c.soft for c in self.cards)

d = Deck()
h = Hand2(d.pop(),d.pop(),d.pop(),d.pop())
print(h.cards)

不带__init__方法的无状态对象

一个策略对象以插件的形式复合在主对象上来完成一种算法或逻辑.它或许以来主对象中的数据,策略对象自身并不携带任何数据.通常策略类会和亨元设计模式一起使用:在策略对象中避免内部存储.所需要的值都从策略对象方法参数传入.策略对象自身是无状态的.可以把它看做是一系列函数的集合.

这里定义了一个类给Player实例提供了游戏的选择模式,以下这个策略包括拿牌和下注.

class GameStrategy:
    def insurnace(self, hand):
        return False

    def split(self, hand):
        return False

    def double(self, hand):
        return False

    def hit(self, hand):
        return False

每个函数需要传入已有的Hand对象,函数逻辑所需要的数据基于现有的可用信息.意味着数据来自于庄家跟玩家的手牌.

一起其他的类定义

玩家有两张策略:打牌和下注.每个Player实例回合模拟器进行很多次交互.我们这里把这个模拟器命名为Table

Table类的职责需要配合Player实例完成以下事件:

  • 玩家必须要基于玩牌策略初始化一个牌局.
  • 随后玩家会得到一手牌
  • 如果

以下是Table类中投注和牌的逻辑处理相关的代码


class Table:
    def __init__(self):
        # 生成52张牌
        self.deck = Deck()

    def place_bet(self, amount):
        print('Bet', amount)

    def get_hand(self):
        try:
            # self.hand = Hand2(d.pop(), d.pop(), d.pop())
            # self.hole_card = d.pop() 书上是这么写的我认为不对,改为下面写法
            self.hand = Hand2(self.deck.pop(), self.deck.pop(), self.deck.pop())
            self.hole_card = self.deck.pop()
        except IndexError:
            # Out of cards: need to shuffle
            self.deck = Deck()
            return self.get_hand()
        print('Deal', self.hand)
        return self.hand

    # 没有看明白hand从何而来,所以也未找到insure的方法。估计是写错了。
    def can_insure(self, hand):
        return hand.dealer_card.insure

class BettingStrategy:
    def bet(self):
        raise NotImplementedError('No bet method')
    
    def record_win(self):
        pass
    
    def record_lose(self):
        pass
    
class Flat(BettingStrategy):
    def bet(self):
        return 1

上面的那一段代码还未看懂需要以后再来看一遍.

多策略的__init__()方法

class Hand4:
    def __init__(self, *args, **kwargs):
        print(len(args),args,kwargs)
        if len(args) == 1 and isinstance(args[0], Hand4):
            other = args[0]
            self.dealer_card = other.dealer_card
            self.cards = other.cards

        elif len(args) == 2 and isinstance(args[0], Hand4) and 'split' in kwargs:
            # Split an existing hand
            other, card = args
            self.dealer_card = other.dealer_card
            self.cards = [other.cards[kwargs['split']], card]
        elif len(args) == 3:
            # Bulid a fresh ,new hand
            dealer_card,*cards = args
            self.dealer_card = dealer_card
            self.cards = list(cards)

        else:
            raise TypeError('Invaild constructor args= {0!r} kw={1!r}'.format(args,kwargs))

    def __str__(self):
        return ','.join(map(str,self.cards))


d = Deck()

h = Hand4(d.pop(),d.pop(),d.pop())
print(h)

# s1 = Hand4(h,d.pop(),split = 0)
# s2 = Hand4(h,d.pop(),split = 1)
class Hand5:
    def __init__(self,dealer_card,*cards):
        self.dealer_card = dealer_card
        self.cards = list(cards)

    @staticmethod
    def freeze(other):
        hand = Hand5(other.dealer_card,*other.cards)
        return hand

    @staticmethod
    def split(other,card0,card1):
        hand0 = Hand5(other.dealer_card,other.cards[0],card0)
        hand1 = Hand5(other.dealer_card,other.cards[1],card1)
        return hand0,hand1

    def __str__(self):
        return ','.join(map(str,self.cards))

d = Deck()

h = Hand5(d.pop(),d.pop(),d.pop())
s1,s2 = Hand5.split(h,d.pop(),d.pop())

上面这段代码实现了:当第一轮发完牌是,dealer手牌有一张,Player手牌有两张,当手牌的两张牌相同的时候玩家可以选择分牌,将手中的的两张牌分为两组牌,继续进行游戏.然后发牌器会给Palyer每组牌中个发一张牌

更多的__init__()技术

一下是Player类的定义,初始化使用两个策略对象和一个table对象

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容