Абстрактный заводской шаблон в Python

Aug 17 2020

Я читал сегодня об абстрактном шаблоне фабрики и попытался реализовать следующую реализацию.

Я видел множество реализаций в Интернете, где используются 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!
>>>

Если есть какой-то аспект фабрик, которые на самом деле делают что-то другое и, следовательно, требуют отдельных производных классов, вам необходимо включить это в свой вопрос.