- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我在想我是否真的需要一个服务层。
我正在将 spring + hibernate 用于桌面 swing 应用程序,此时我有 gui/swing 层->服务层->dao 层。我仅将 spring 用于@Transactional 支持和 IOC 注入(inject)
最佳实践是我必须编写一个服务来使用我的 daos,并将所有事务管理放在该服务中。
但我经常意识到,服务层只复制 dao 方法,例如:
// a DAO example
@Repository
public class CustomerHibernateDAO extends BaseHibernateDAO implements CustomerDAO {
public List<Customer> findAllCustomerILikeName(String name){
return getSession()
.createCriteria(Customer.class)
.add(Restriction.ilike("name", name))
.list();
}
}
// Customer service to use this dao...
@Service
@Transactional
public class CustomerService {
@Autowired
CustomerDAO customerDAO;
// Why i can't call DAO instead the service?
public List<Customer> getAllCustomersByName(String name){
return customerDAO.findAllCustomerILikeName(name);
}
}
这是服务层的典型用法...Hibernate 与数据库无关,而 spring 与技术无关:那么,我真的需要它吗?
管理所有 DAO 的唯一服务类怎么样?我认为这可能是一个很好的妥协,还是一种不好的做法?
我知道将 @Transactional 放在 DAO 上是一种糟糕的方式,但此时我必须编写仅用于将 @Transactional 放在其上的服务...
编辑
有关我的应用的更多信息。
我的应用程序是一个管理软件,可以管理用户注册、产品、订单和其他类似的东西。在实践中,它包含很多读取实体->编辑->保存实体或创建->编辑->保存操作,而且,由于 hibernate,这些操作大部分时间都由 ONE dao 管理,因为 hibernate 与@manyto。 .. collection 和 cascade.save_update 允许在同一个持久化操作中保存两个或多个实体。
因此,例如,在我可以插入、编辑或创建项目(要销售的产品)的项目 JFrame 中,有:
public ItemFrame(){
// the constructor
itemService=springAppContext.getBeans(ItemService.class);
}
public boolean validateForm(){
// test if the gui is correctly filled by user
}
public boolean save(){
// create an Item entity taking value from swing gui(JTextField etc)
Item item=new Item();
item.setName(nameTextField.getText());
item.setEtc...
// ItemService ' save method is a wrap around itemDao.save(item)...
itemService.save(item);
}
private void saveItemActionPerformed(ActionEvent evt){
// When i press SAVE button
if(validateForm()){
save();
}
}
这是我在大多数情况下遇到的情况,所以我认为我陷入了贫血领域反模式......
谢谢。
最佳答案
如果你的服务层复制了 dao,那么你根本就没有在使用服务层。我在我的几个应用程序中犯了同样的错误,我想知道“为什么服务层看起来这么丑,而且是 duplicationg DAO”...
服务层应该是你的应用程序的接口(interface),这并不意味着,dao 和服务中的一些方法是不一样的,但主要部分是显着不同的。如果不查看您的其余代码,我不能这么说,但是根据您的问题(这与我几个月前的问题几乎相同),在我看来,您正在使用 anemic domain model antipattern .在贫血领域模型中,您的模型仅包含字段和 setter/getter ,没有真正的方法(行为),这违反了基本的面向对象原则(对象==数据+行为)...您的行为可能看起来像服务中的事务脚本层,但应该在您的模型(域层)中。
解决这个问题的方法是使用丰富的域模型(通过@Configurable 将 bean 注入(inject)到模型中)。你可能会说,这违反了图层模式,你可能是对的。但我坚信,我们应该将我们的应用程序(域+dao+服务)视为一个单独的组件(参见 Alistair Cockburn Hexagonal architecture/Ports and adapters)。
然后,您的 swing 应用程序/网络客户端将成为核心组件的客户端,然后您可以无限制地切换它们(因为修改数据的所有内容都在核心中)。
但是这种方法有一个限制/缺点。如果您将通过 hibernate 使用某种安全性(Spring 安全性)或 Activity 记录,那么您应该通过 DTO(而不是实体本身)与所有客户进行通信,因为当您联系实体时,它可能会调用服务,该服务将通过事务性的并且可以修改数据库(绕过您的安全)。
我希望,我已经猜到了你的架构,如果没有,我很抱歉在这里发明了轮子,但这篇文章可能会帮助那些不知道这一点的人(就像我几个月前一样)。
编辑
您的编辑:即使在简单的 CRUD 应用程序中,某些类型的操作也应该在服务层中 - 例如验证(不是验证“这是一个数字”,而是一些特定于业务的验证)。这应该在您的 View 中,因为如果您更改它,您将再次复制并粘贴它。当你查看你的代码时,你应该问一个问题“如果我决定编写瘦客户端(在网络浏览器中查看)”,是否有任何代码,我必须复制?如果答案是肯定的,那么您应该为这个可能的远程调用创建一个服务方法。
您应该/可以在服务层做的另一件事是授权(是否允许该角色的用户删除该条目)。比起你几乎所有的实体都必须有一个服务层,因为简单的用户应该能够编辑(删除)他的条目,但可能不应该删除其他用户。但是管理员角色的用户可以执行此操作。
示例代码(文章服务接口(interface)在我的应用程序(Spring 安全)中的一部分):
@Secured("ROLE_EDITOR")
public void save(ArticleDTO selectedArticle, ArticleDetailsDTO selectedArticleDetails);
在评论服务中,每个人都可以将他们对文章的评论保存....
最后一点:如果您需要服务层,您可能应该考虑一下。如果以良好的方式编写,您的应用程序将在灵 active 、可重用性和可维护性方面获得许多品质。但是编写它非常困难且耗时。如果你不想做所有这些事情(安全、丰富的领域模型、从更多接口(interface)调用(改变 View 实现)),你可以没有它:-)
关于java - java swing应用程序中的服务层,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4270461/
我正在开发一个摆动程序以显示多张图片。并且可以旋转图片(每个图片都以JComponent实现)。 问题是,当图片旋转时,JComponent的边框不会改变,因此图片会被剪切。 有什么办法也可以旋转边框
我有一个 JPanel 和一个 JButton 向量,我想将每个按钮添加到面板。 我遇到的问题是我有一个代表按钮向量的变量 btns,但宏函数只是将它视为一个符号,而不是一个向量。有没有办法以某种方式
我有一个 swing 应用程序,它覆盖 javax.swing.text.Document 以将基础文档的内容限制为某些字符和文本长度。我想将我的应用程序移植到 Javafx,但我不知道此类是否有 J
我有一个带有面板的简单应用程序,我想在单击它时暂停并重新开始绘画。 object ModulusPatterns extends SimpleSwingApplication { var dela
我似乎无法在 Swing 中强制布局。我有一个 JComponent添加到 JLayeredPane我在 JComponent 上设置了边框.然后,我想立即重新绘制所有内容 - 而不是像 invali
我有一个 Swing 应用程序,我想通过将外部文件从 Windows 资源管理器拖到应用程序上来导入外部文件。我有这个基本功能工作。但是,我想将默认的拖放光标图标更改为应用程序适当的光标。当鼠标键被按
我想在我的 Scala 摆动应用程序中使用一棵树,但该组件在 API 中不可用。 是否包装了 JTree存在吗? 如果没有,你对制作有什么建议吗? 谢谢 最佳答案 即使您可以在 Scala 程序中直接
我目前正在努力使我的 Swing 应用程序看起来更好。我想在这些方面实现一些目标: 这个想法是让每个框都有一个漂亮的标题,背景类似于上图。使用基本的 Swing 组件,我能得到的最接近的东西是添加 T
这是我在 Scala 中使用 Swing 的第一次实验,并且对下面的代码有一些疑问。它所做的只是生成一个带有可改变颜色的彩色矩形的窗口。请随时回答一个或任何一个问题。 1) 我在下面使用了 Java
一个愚蠢的问题,但我真的无法让它起作用:我在 Swing 应用程序中有一些长时间运行的过程,可能需要几分钟。我想在此过程进行时向用户显示进度对话框。我还想阻止用户执行进一步的操作,例如在进程进行时按下
如何获取秋千组件的默认背景色?我的意思是JPanel的默认背景色? 最佳答案 要获取创建面板时将使用的 DEFAULT 颜色,请使用: Color color = UIManager.getColor
我想更改特定表头的背景颜色。在我的应用程序中,我必须将当前月份的标题颜色设置为红色。 我的代码在这里:: jTable1.getTableHeader(). setDefaultRe
我正在努力使在 Java3D Canvas 上显示 Java Swing 组件并与之交互成为可能。我通过将透明 JPanel 绘制到缓冲图像来显示组件,然后使用 J3DGraphics2D 在 Can
嗨 当我在 swing 中创建按钮时,它会在文本周围添加边框,从而使按钮变大一点。 现在,我确实需要那个屏幕空间,我通常做的是创建一个文本项(禁用),它创建更小的组件大小(文本周围更小的空间)并向其添
有scala.swing.BoxPanel,但它似乎没有捕获重点,因为没有与javax.swing.Box工厂方法createHorizontalStrut等效的东西、createHorizo
我的 scala swing 应用程序中有一个 BoxPanel 按钮,这对我来说很难看,因为按钮的大小各不相同。我已将其更改为 GridPanel,但随后它们也垂直填充了面板,我发现这更难看。我怎样
我刚开始学习 Scala,作为学习过程的一部分,我一直在尝试使用 Swing 编写一些简单的脚本。 这是一个非常精简的例子,它展示了我所看到的问题。 SimpleSwingApp: import sc
我刚刚开始使用 clojure 和跷跷板制作 GUI 应用程序。它只创建一个 JFrame 和几个组件。这是代码。 main 函数除了调用 start-gui 什么也不做并在返回后立即退出。 (ns
Scala 是一种很棒的语言,但不幸的是缺少库文档。如何更改组件的初始大小?我什么都没有(故意),但无论如何都希望它是一定的。我目前有 ... contents = new BoxPanel(Orie
基本设置是这样的:我有一个垂直的 JSplitPane,我希望它有一个固定大小的底部组件和一个调整大小的顶部组件,我通过调用 setResizeWeight(1.0) 来完成。在此应用程序中,有一个按
我是一名优秀的程序员,十分优秀!