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