3

我正在为网站编写功能区/成就系统,我必须为系统中的每个功能区编写一些逻辑。例如,如果您是注册网站的前 2,000 人中或在论坛中发帖 1,000 条之后,您就可以获得勋章。这个想法与stackoverflow的徽章非常相似,真的。

因此,每个功能区显然都在数据库中,但它们还需要一些逻辑来确定用户何时获得功能区。

以我编码的方式,Ribbon是一个简单的界面:

public interface Ribbon {
    public void setId(int id);
    public int getId();
    public String getTitle();
    public void setTitle(String title);
    public boolean isEarned(User user);
}

RibbonJpa是实现Ribbon接口的抽象类,避免了isEarned()方法的定义:

@Entity
@Table(name = "ribbon")
@Inheritance(strategy = InheritanceType.SINGLE_TABLE)
@DiscriminatorColumn(name = "ribbon_type")
public abstract class RibbonJpa implements Ribbon {
    @Id
    @Column(name = "id", nullable = false)
    int id;

    @Column(name = "title", nullable = false)
    private String title;

    @Override
    public int getId() {
        return id;
    }

    @Override
    public void setId(int id) {
        this.id= id;
    }

    @Override
    public String getTitle() {
        return title;
    }

    @Override
    public void setTitle(String title) {
        this.title = title;
    }
}

您可以看到我将继承策略定义为SINGLE_TABLE(因为我必须编写 50 个功能区之类的代码,并且我不需要为它们中的任何一个添加额外的列)。

现在,一个特定的功能区将像这样实现:

@Entity
public class FirstUsersRibbon extends RibbonJpa implements Ribbon {
    public FirstUsersRibbon() {
        super.setId(1);
        super.setTitle("First 2,000 users registered to the website");
    }

    @Override
    public boolean isEarned(User user) {
        // My logic to check whether the specified user has won the award
    }
}

这段代码工作正常,表以我期望的方式在数据库中创建(我在本地环境中使用 DDL 生成)。

问题是,在域对象中编写业务逻辑感觉是错误的。这是好习惯吗?您能提出更好的解决方案吗?此外,我无法自动装配实体 ( FirstUsersRibbon) 中的任何 DAO,我需要在业务逻辑中使用它们(在这种情况下,我需要一个 DAO 来检查用户是否在注册到网站的前 2,000 个用户中)。

非常感谢任何帮助。

谢谢!

4

2 回答 2

11

问题是,在域对象中编写业务逻辑感觉是错误的。

许多人会说相反的情况是正确的:在其他任何地方都有业务逻辑是一种反模式(一种贫血的领域模型)。有关详细信息,请参阅领域驱动设计

然后您可能想知道传统 3 层架构的中间层是干什么用的。它为应用程序提供了一个服务层。请参阅我的相关问题“ EJB 有什么用途? ”。

于 2012-05-20T19:41:13.013 回答
0

此外,我无法自动装配实体中的任何 DAO

如果您使用 Spring 和 Hibernate,请查看http://jblewitt.com/blog/?p=129:这很好地描述了具有各种解决方案的类似问题。

如果您正在以您描述的方式寻找丰富的域模型,那么通过 Spring 实例化域对象可能是一个好主意,因此能够将 DAO 注入到您的域对象中。

于 2012-05-20T20:23:26.983 回答