我们今天聊一下 Spring Data JPA
里的三个方法,分别是 getOne
, findOne
以及 findById
。
咋一看三个方法都能返回一个结果集,用哪个好像都没问题。我当初也是这么想的,后来在写作业的过程中出错了,真相只有一个。
我的例子是查询一个一对一映射关系的实体,极为简单,我就不上代码了;用 findById
则可以实现我们的需求,而使用 getOne
查询后对结果集进行打印,出现下面的异常。
org.hibernate.LazyInitializationException: could not initialize proxy - no Session
这个异常好像跟使用哪个方法没太大的关系,因为他是在得到查询的结果集后,打印这个结果集的时候出现的,且映射关系默认使用 FetchType.EAGER
,实体的 toString
方法仅对实体自己的属性做打印,不处理映射关系,莫非是因为得到的一个代理对象,不能映射成实体?还希望路过的高人指点一二。
最后使用 findOne
,也能准确的查到结果,有趣的是打印了三条查询语句。
翻开官方的 API ,找找这几个磨人的方法都在哪:
getOne
来自 JpaReposiroty
接口,对于传入的标识则返回一个实体的引用;且取决于该方法的实现,可能会出现 EntityNotFoundException
,并会拒绝一些无效的标识;
findById
来自 CrudRepository
接口,通过它的 id
返回一个实体;
findOne
来自 QueryByExampleExecutor
接口,返回一个通过 Example
匹配的实体或者 null
;
那他们的区别也就是:
getOne
返回一个实体的引用,无结果会抛出异常; findById
返回一个 Optional
对象; findOne
返回一个 Optional
对象,可以实现动态查询; 注:文中使用的版本为 Spring Data JPA 2.1.8.RELEASE
。