Spring开发人员知道的一件事

在最近关于(核心)Spring Framework的培训课程中,有人问我:“(Java)Spring开发人员是否应该知道一件事,那应该是什么?” 这个问题使我措手不及。 是的,(核心)Spring框架确实涵盖了很多领域(例如,bean,配置,面向方面的编程,事务)。 我很难指出一件事。 最后,我提到了我们为期3天的培训课程所涵盖的所有内容。

如果(Java)Spring开发人员应该知道的一件事,那应该是什么?

当我进一步思考这个问题时,我开始思考重要的一个问题。 最后,我想到了Spring如何重要地使用方面将行为添加到托管对象(通常称为bean)中。 Spring Framework通过这种方式来支持事务,安全性,范围,基于Java的配置等。 我在这篇文章中分享我的想法。

最后,我想到了Spring如何重要地使用方面将行为添加到托管对象(通常称为bean)中。

ORM和延迟加载异常

大多数使用某种形式的ORM的开发人员都遇到了一个异常,该异常表明无法加载子实体(例如LazyInitializationException )。

一些遇到这种情况的开发人员将使用“打开的视图中的会话”( OSIV )模式来保持会话打开并防止发生此异常。 但是我觉得这太过分了。 更糟糕的是,一些开发人员认为“开放会话可见”模式是唯一的解决方案。 造成这种误解的潜在根本原因可能是,开发人员可能不了解有效使用Spring Framework来保持ORM会话打开时间更长的知识。

对于JPA ,“打开视图中的实体管理器”模式将在请求开始时创建一个实体管理器,将其绑定到请求线程,并在响应完成时将其关闭。

那么,如果不是OSIV模式,哪种解决方案更好?

简短的答案是使用Spring框架在需要的时间内保持会话打开(例如@Transactional )。 请继续阅读,因为我会提供更长的答案。

服务和存储库

在分层体系结构中,典型的设计模式是定义域或应用程序服务 (通常定义为接口)以提供业务功能(例如,开始使用购物车,向该购物车添加商品,搜索产品)。 域和应用程序服务的实现通常会将域实体的检索/持久性委派给存储库。

One_Thing_Good_Spring_Developers_Know

存储库 (或数据访问对象)也被定义为检索/持久化域实体(即提供ORM和CRUD访问)的接口。 自然地,存储库实现使用ORM库(例如JPA / Hibernate,myBATIS)来检索和持久化域实体。 这样,它使用ORM框架的类连接到持久性存储,检索/持久化实体并关闭连接(在Hibernate中称为会话)。 此时,没有延迟加载失败的问题。

当服务使用资料库中检索域实体,并希望加载的子实体(库方法返回之后 )发生的懒加载失败问题。 到存储库返回域实体时,ORM会话将关闭。 因此,尝试访问/加载域服务中的子实体会导致异常。

下面的代码段说明了当订单实体的子项目由存储库返回后被延迟加载时,如何发生延迟加载异常。

@Entity
public class Order {@OneToMany // defaults to FetchType.LAZYprivate List<OrderItem> items;…public List<OrderItem> getItems() {…}
}public class SomeApplicationServiceImpl implements SomeApplicationService {private OrderRepository orderRepository;…@Overridepublic void method1(…) {…order = orderRepository.findById(...);order.getItems(); // <-- Lazy loading exception occurs!…}…
}public class OrderRepositoryImpl implements OrderRepository {@PersistenceContextprivate EntityManager em;…@Overridepublic Order findById(...) {...}…
}

存储库实现将JPA明确用于其ORM(如使用EntityManager )。

在这一点上,一些开发人员可能选择使用紧急获取来防止延迟初始化异常。 告诉ORM急切地获取订单实体的子项将起作用。 但是有时候,我们不需要加载子项。 急于加载它可能是不必要的开销。 仅在需要时加载它会很棒。

为了防止延迟初始化异常(而不是被迫急于获取),我们需要保持ORM会话打开,直到调用服务方法返回。 在Spring中,可以像@Transactional一样简单地注释服务方法以保持会话打开。 我发现这种方法比使用“在视图中打开会话”模式(或被迫使用急切获取)更好,因为它使会话仅在我们希望的时间内保持打开状态。

public class SomeApplicationServiceImpl implements SomeApplicationService {private OrderRepository orderRepository;…@Override@Transactional // <-- open the session (if it's not yet open)public void method1(…) {…order = orderRepository.findById(...);order.getItems(); // <-- Lazy loading exception should not happen…}…
}

表示层中的域实体

即使将ORM会话在服务层中(在存储库实现对象之外)保持打开状态,当我们将域实体暴露给表示层时,仍然可能发生惰性初始化异常。 同样,由于这个原因,一些开发人员更喜欢OSIV方法,因为它还可以防止表示层中的延迟初始化异常。

但是,为什么要在表示层中公开域实体?

根据经验,我曾与那些希望在表示层中公开域实体的团队合作。 这通常会导致贫血领域模型 ,因为表示层框架需要一种将输入值绑定到对象的方法。 这迫使域实体具有getter和setter方法以及零参数构造函数。 具有getter和setter将使不变式难以执行。 对于简单域,这是可行的。 但是对于更复杂的领域,更丰富的领域模型将是首选,因为它更易于实施不变式。

在更丰富的域模型中,表示表示层输入/输出值的对象实际上是数据传输对象(DTO)。 它们代表在域层中执行的输入(或命令)。 考虑到这一点,我更喜欢使用DTO并维护更丰富的域模型。 因此,我并没有真正在表示层遇到懒惰的初始化异常。

向受管对象添加行为的方面

Spring会拦截对这些@Transactional注释方法的调用,以确保ORM会话处于打开状态。

事务(或只是保持ORM会话保持打开状态)并不是使用方面提供的唯一行为。 有安全性,范围,基于Java的配置等。 知道Spring框架使用方面来添加行为是我们让Spring管理我们开发的POJO的关键原因之一。

结论

妳去 对我来说,这是Spring Framework开发人员在使用内核时应了解的最重要件事。 现在,我已经对什么是最重要的事情发表了意见,您呢? 在处理Core Spring时,您认为最重要件事是。 干杯!

翻译自: https://www.javacodegeeks.com/2016/02/one-thing-good-spring-developers-know.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/354993.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql匿名事务gtid_MySQL GTID (二)

MySQL GTID 系列之二三.在线将GTID转化为传统模式环境见上篇系列文章关闭GTID,不用停止服务,不影响线上业务3.1 关闭GTID复制,调整为传统复制#SLVAE实例上停止复制STOP SLAVE#SLVAE实例上查看复制的位置SHOW SLAVE STATUS \G# 查看 Master_Log_File 和 Read_Master_Log_Pos对应…

JavaWeb(十七)——JSP中的九个内置对象

一、JSP运行原理 每个JSP 页面在第一次被访问时&#xff0c;WEB容器都会把请求交给JSP引擎&#xff08;即一个Java程序&#xff09;去处理。JSP引擎先将JSP翻译成一个_jspServlet(实质上也是一个servlet) &#xff0c;然后按照servlet的调用方式进行调用。  由于JSP第一次访问…

jrockit_JRockit – JRCMD有用的命令

jrockit自2007年以来&#xff0c;我一直在使用JRockit。我发现它的速度比Hotspot慢&#xff0c;但在诊断和分析问题上总是更好。 从去年夏天开始&#xff0c;我一直在为一家国际电信系统供应商工作。 我们在HP OpenCall融合通信平台之上为电信运营商设计和实施各种产品。 我是开…

java并发问题_并发理论基础:并发问题产生的三大根源

并发问题变幻莫测&#xff0c;一谈到并发就显得非常高深&#xff0c;一般的程序员对于并发问题也是头疼不已&#xff0c;但是随着网络互联越来越普遍&#xff0c;大规模用户访问网站程序也越来越频繁&#xff0c;并发问题又无法避免。在我们解决并发问题前首先要理解产生并发问…

[luoguP1849] [USACO12MAR]拖拉机Tractor(spfa)

传送门 神奇的spfa #include <queue> #include <cstdio> #include <cstring> #include <iostream> #define N 1010 #define max(x, y) ((x) > (y) ? (x) : (y))int n, mx, my; int dis[N][N]; bool map[N][N], vis[N][N]; int dx[4] {0, -1, 0, 1…

在Eclipse上创建JSF / CDI Maven项目

当我在研究JSF和CDI示例时&#xff0c;我认为提及创建JSF和CDI Maven项目所需的步骤会很有用。 您可以找到以下步骤。 工具类 默认情况下&#xff0c;M2E插件随附的Eclipse Luna。 因此&#xff0c;无需自己安装插件。 WildFlye8.x。 从主菜单中选择文件->新建->其他。…

luoguP3690 【模板】Link Cut Tree (动态树)[LCT]

题目背景 动态树 题目描述 给定&#xff2e;个点以及每个点的权值&#xff0c;要你处理接下来的&#xff2d;个操作。操作有&#xff14;种。操作从&#xff10;到&#xff13;编号。点从&#xff11;到&#xff2e;编号。 &#xff10;&#xff1a;后接两个整数&#xff08;&a…

python爬虫多进程_Python爬虫技术--基础篇--多进程

要让Python程序实现多进程(multiprocessing)&#xff0c;我们先了解操作系统的相关知识。Unix/Linux操作系统提供了一个fork()系统调用&#xff0c;它非常特殊。普通的函数调用&#xff0c;调用一次&#xff0c;返回一次&#xff0c;但是fork()调用一次&#xff0c;返回两次&am…

java 电力系统_算法java实现--动态规划--电路布线问题

/** dianlubuxian.java* Version 1.0.0* Created on 2017年11月30日* Copyright ReYo.Cn*/package reyo.sdk.utils.test.dy;/*** 创 建 人&#xff1a;AdministratorReyoAut * 创建时间&#xff1a;2017年11月30日 下午4:58:56** author ReYo* version 1.0*//*** 电路布线问题(…

百度图片网址

http://qcloud.dpfile.com/pc/jPAgaVMWC7zueHYEzky7IUJs0w6QIgvTQ0p08wxCK1OUDUk6-KqvLg70OVUXtjEHTYGVDmosZWTLal1WbWRW3A.jpg转载于:https://www.cnblogs.com/leshen/p/7387677.html

antlr idea 入门_ANTLR:入门

antlr idea 入门这篇文章使您了解ANTLR的基础知识。 以前&#xff0c;我们已经了解了如何将ANTLR设置为外部工具。 在这里&#xff1a; ANTLR外部工具 :) 所以&#xff0c;我们开始…。 什么是ANTLR&#xff1f; •另一个语言识别工具&#xff0c;是一种语言工具&#xff0c;它…

typescript主键自增长

常见的不重复id创建方式有两种&#xff0c;一个是搞一个自增长数列&#xff0c;另一个是采用随机生成一组不可能重复的字符序列&#xff0c;常见的就是UUID了。我们来引入一个uuid的包&#xff1a;npm i --save angular2-uuid&#xff0c;由于这个包中已经含有了用于typescript…

java api操作hbase_通过JavaAPI使用HBase

1.准备工作(1) 启动zookeeper服务&#xff0c;我的是在本地启动zookeeper/usr/local/zookeeper/bin$ sudo zkServer.sh start(2) 启动HBase和HBase shell启动HBase:/usr/local/hbase/bin下启动start-hbase.sh启动HBase shell/usr/local/hbase/bin下终端输入hbase shell(3) 工程…

SPOJ QTREE5 lct

题目链接 对于每一个节点&#xff0c;记录这个节点所在链的信息&#xff1a; ls:&#xff08;链的上端点&#xff09;距离链内部近期的白点距离 rs:&#xff08;链的下端点&#xff09;距离链内部近期的白点距离 注意以上都是实边 虚边的信息用一个set维护。 set维护的是…

Java EE 8 MVC:使用路径参数

在上一篇文章中&#xff0c;我们看到了如何在Java EE MVC中使用查询参数 。 这篇文章继续与一个非常相似的主题&#xff1a;路径参数。 路径参数是请求路径的动态部分&#xff0c;可以使用Path注释指定。 例如&#xff1a; Controller Path("path-params") public…

duilib入门简明教程 -- 部分bug (11) (转)

原文转自&#xff1a;http://www.cnblogs.com/Alberl/p/3344886.html 一、WindowImplBase的bug在第8个教程【2013 duilib入门简明教程 -- 完整的自绘标题栏(8)】中&#xff0c;可以发现窗口最大化之后有两个问题&#xff0c;1、最大化按钮的样式还是没变&#xff0c;正确的样式…

在考生文件夹存有JAVA3_注意:下面出现的“考生文件夹”均为%USER%在考生文件夹下存有文件名为J_网考网(Netkao.com)...

【分析解答题】注意&#xff1a;下面出现的“考生文件夹”均为%USER%在考生文件夹下存有文件名为Java_2.java文件&#xff0c;本题功能是完成点定义&#xff0c;并输出点坐标。请完善Java_2.java文件&#xff0e;并进行调试&#xff0c;使程序结果如下&#xff1a;x5 y5点的坐标…

jasperreports_JasperReports JSF插件用例系列

jasperreports这是文章系列的切入点&#xff0c;在该系列文章中&#xff0c;我将尝试介绍JasperReport JSF插件的一些用例&#xff0c;该工具的创建是为了轻松地将为JasperReports设计的业务报告集成到JSF应用程序中。 该系列中描述的所有示例都可以从JasperReports JSF插件网站…

RN 47 中的 JS 线程及 RunLoop

RCBridge 初始化时声明了一个 CADisplayLink _jsDisplayLink [CADisplayLink displayLinkWithTarget:self selector:selector(_jsThreadUpdate:)];在 _jsThreadUpdate 函数中&#xff0c;处理界面更新。这个 CADisplayLink 随后被加到 JS 线程对应的 RunLoop 中。 - (void)ad…

java nginx https_docker nginx 配置ssl,实现https

docker nginx 配置ssl&#xff0c;实现https2019-09-05 16:06:35.0nginx配置https总览在nginx配置ssl实现https&#xff0c;简单来说分为三个步骤&#xff1a;1 上传ssl证书等文件将 1_www.domain.com_bundle.crt 和 2_www.domain.com.key 上传到nginx配置文件的目录旁边。这两…