Управление объектами стратегии с помощью Hibernate & Spring - PullRequest
2 голосов
/ 28 мая 2010

Этот вопрос проектирования лучше объяснить с помощью аналогии с переполнением стека:

Пользователи могут зарабатывать значки. Пользователи, значки и заработанные значки хранятся в базе данных. Логика бейджа управляется стратегией состояния бейджа. Я бы предпочел не хранить стратегии состояний значков в базе данных, поскольку они представляют собой сложные объекты древовидной структуры.

Как связать значок, хранящийся в базе данных, с его стратегией состояния значка? Я могу думать только об обходных решениях. Например: создайте 1 класс на значок и используйте стратегию наследования SINGLE_TABLE. Или получите значок из базы данных, а затем программно найдите и введите правильную стратегию состояния значка.

Спасибо, что предложили лучший дизайн.

Ответы [ 2 ]

1 голос
/ 28 мая 2010

Как насчет перечисления BadgeType с типами, которые соответствуют значкам в базе данных? Перечисление может иметь метод getBadgeConditionStrategy (), который возвращает правильную стратегию для каждого значения перечисления:

public enum BadgeType {
     SMARTNESS( new SmartnessBadgeConditionStrategy() ),
     WISDOM( new WisdomBadgeConditionStrategy(),
     ...;

     private BadgeConditionStrategy badgeConditionStrategy;
     BadgeType(BadgeConditionStrategy badgeConditionStrategy) {
          this.badgeConditionStrategy = badgeConditionStrategy;
     }

     public getBadgeConditionStrategy() {
         return badgeConditionStrategy;
     }
 }
1 голос
/ 28 мая 2010

Я не понимаю, почему вы должны хранить стратегию в БД - стратегия в основном выражается в коде (возможно, с некоторыми параметрами конфигурации, которые, в свою очередь, могут храниться в БД, но для меня это другая проблема).

OTOH Я бы держал значок и его стратегию состояния в одном классе, что устраняет вашу проблему с поиском. В Java хорошей моделью предметной области было бы представление значков в виде перечисления с переопределенным методом, чтобы определить, заработал ли данный пользователь этот конкретный значок.

Обновление Вот пример:

enum Badge {
  EPIC() {
    public boolean isEligible(User user) {
      // determine whether this user is eligible for the Epic badge
    }
  },
  CRITIC() {
    public boolean isEligible(User user) {
      // determine whether this user is eligible for the Critic badge
    }
  },
  ...
  ;

  public abstract boolean isEligible(User user);
}

Но если вы действительно хотите их разделить, то в конструкторе, например, LegendaryBadge Вы говорите this.strategy = new LegendaryBadgeConditionStrategy();

...