Pythonの抽象ファクトリパターン

Aug 17 2020

今日はAbstractFactory Patternについて読んでいて、次の実装を試みました。

インターネットでswitchステートメントを使用する実装をたくさん見ましたが、工場が増えるほど新製品の追加が非常に難しくなるので、あまり好きではなかったと言わざるを得ません。必要に応じて。

とにかく、ぜひご覧いただき、ご意見をお聞かせください。時間をかけてレビューしていただき、ありがとうございます。

工場

from abc import ABC, abstractmethod


class PlayerFactory(ABC):
    """
    This class is meant to be an interface
    """
    @abstractmethod
    def create_goalkeeper(self):
        pass

    @abstractmethod
    def create_defender(self):
        pass


class FootballPlayerFactory(PlayerFactory):
    def create_goalkeeper(self):
        return FootballGoalkeeper()

    def create_defender(self):
        return FootballDefender()


class HockeyPlayerFactory(PlayerFactory):
    def create_goalkeeper(self):
        return HockeyGoalkeeper()

    def create_defender(self):
        return HockeyDefender()

サッカー選手

class FootballPlayer:
    def __init__(self, uses_hands):
        self.uses_hands = uses_hands

    def play(self):
        print("I'm playing football!")


class FootballGoalkeeper(FootballPlayer):
    def __init__(self):
        super(FootballGoalkeeper, self).__init__(uses_hands=True)


class FootballDefender(FootballPlayer):
    def __init__(self):
        super(FootballDefender, self).__init__(uses_hands=False)

ホッケー選手私の創造性はここで止まったので、ゴールキーパーとディフェンダーの違いは含めませんでした

class HockeyPlayer:
    def play(self):
        print("I'm playing hockey!")


class HockeyGoalkeeper(HockeyPlayer):
    pass


class HockeyDefender(HockeyPlayer):
    pass

回答

2 AJNeufeld Aug 17 2020 at 22:24

現在のコードでは、派生したFactoryクラスは必要ありません。それらは互いに何も変わらないので、すべて具体的な基本クラスで処理できます。

class PlayerFactory:

    def __init__(self, goal_keeper_class, defender_class):
        self._goal_keeper_class = goal_keeper_class
        self._defender_class = defender_class

    def create_goalkeeper(self):
        return self._goal_keeper_class()

    def create_defender(self):
        return self._defender_class()

player_factory = {
    "Football": PlayerFactory(FootballGoalkeeper, FootballDefender),
    "Hockey": PlayerFactory(HockeyGoalkeeper, HockeyDefender),
}

使用例:

>>> player = player_factory["Hockey"].create_defender()
>>> type(player)
<class '__main__.HockeyDefender'>
>>> player.play()
I'm playing hockey!
>>>

実際に何か違うことをするファクトリーのいくつかの側面があり、したがって分離された派生クラスが必要な場合は、それを質問に含める必要があります。