当前位置: 首页 > 面试题库 >

CrudRepository findOne()和JpaRepository getOne()之间的区别

常宸
2023-03-14
问题内容

我读到它getOne()是延迟加载的,并立即findOne()获取整个实体。我检查调试日志,我甚至能监视我的SQL服务器上看到的语句被执行的东西,我发现,无论getOne()findOne()生成和执行相同的查询。但是,当我使用时,getOne()这些值最初为null(当然,除了id)。

因此,谁能告诉我,如果两种方法都在数据库上执行相同的查询,那为什么我要在另一种方法上使用呢?我基本上是在寻找一种获取实体而不获取其所有子代/属性的方法。

编辑1:

实体代码

道代码:

@Repository
public interface FlightDao extends JpaRepository<Flight, Long> {
}

调试日志findOne()vs getOne()

编辑2:

感谢Chlebik,我得以找到问题所在。如Chlebik所述,如果您尝试访问由getOne()完整查询获取的实体的任何属性,则将执行该查询。就我而言,我在调试时检查行为,一次只移动一行,但我完全忘记了在调试IDE时出于调试目的而尝试访问对象属性(或者至少是我认为的那样),因此调试触发器完整的查询执行。我停止调试,然后检查了日志,一切似乎都正常。

getOne()vs findOne()(此日志取自MySQL general_log,不hibernate。

调试日志

没有调试日志


问题答案:

只是一个猜测,但是在“纯JPA”中,有一个EntityManager的方法称为 getReference
。它旨在检索其中仅包含ID的实体。它的使用主要是为了指示引用的存在,而无需检索整个实体。也许代码可以告诉更多信息:

// em is EntityManager
Department dept = em.getReference(Department.class, 30);  // Gets only     entity with ID property, rest is null
Employee emp = new Employee();
emp.setId(53);
emp.setName("Peter");
emp.setDepartment(dept);
dept.getEmployees().add(emp);
em.persist(emp);

我假设 getOne 可以达到相同的目的。为什么生成的查询与您询问的相同?好吧,JPA圣经中的 AFAIR-Mike Keith和MerrickSchincariol编写的Pro JPA2- 几乎每个段落都包含类似“行为取决于卖方”的内容。

编辑:

我已经设定好自己的设定。最后,我得出的结论是,如果您以任何方式干扰使用 getOne
提取的实体(甚至转到entity.getId()),它将导致SQL被执行。尽管如果仅使用它来创建代理(例如,如上面代码中所示的关系指示符),则什么也不会发生,也不会执行其他SQL。因此,我假设在您的服务类中您对此实体进行了某些操作(使用getter进行记录),这就是为什么这两种方法的输出看起来相同的原因。

具有示例代码的ChlebikGitHub
有用的问题#1
有用的问题#2



 类似资料:
  • 问题内容: 我错放了太多次了,我想我一直忘记,因为我不知道两者之间的区别,只是一个给了我我期望的价值,而另一个却没有。 为什么是这样? 问题答案: 是的简写形式(尽管请注意,该表达式只会被计算一次。) 是的,即指定一元的到。 例子:

  • 问题内容: 因此,我有一段简单的代码可以打印出整数1-10: 然后,如果仅在第3行上更改一个运算符,它将打印出无限数量的1整数(我知道为什么会这样做)。为什么在运行第二个程序时没有出现语法错误?如果赋值运算符后面跟着一个加法运算符,它不会调用语法错误吗? 问题答案: 与相同, 只是意味着。

  • 问题内容: 有人可以解释一下 和 我不知道“确切”的含义 问题答案: 在这个例子中,什么都没有。当您具有多个具有相似名称的路径时,该参数将起作用: 例如,假设我们有一个显示用户列表的组件。我们还有一个用于创建用户的组件。的网址应嵌套在下。因此,我们的设置可能如下所示: 现在,这里的问题是,当我们转到路由器时,将通过所有定义的路由,并返回它找到的第一个匹配项。因此,在这种情况下,它将首先找到路线,然

  • 问题内容: 我很好奇printStackTrace()和toString()之间的区别是什么。乍一看,他们 似乎 做的完全相同。 码: 问题答案: 不,有重要区别!使用toString,您只有异常的类型和错误消息。使用printStackTrace()可以获得异常的整个堆栈跟踪,这对于调试非常有帮助。 System.out.println(toString())的示例: printStackTra

  • 问题内容: 我看不到两种方式之间的任何区别,@ Qualifier 始终与 @Autowired一起使用 。 VS 有人能让我知道其中的区别吗?谢谢! 问题答案: 可以单独使用。如果单独使用,将按类型进行接线。因此,如果在容器中声明了多个相同类型的bean,而又不知道要注入哪个bean,就会出现问题。结果,通过指定Bean名称(按名称进行绑定),与一起使用来阐明要实际连接的Bean 也按名称接线。

  • 问题内容: System.getenv()和System.getProperty()有什么区别? 当我们使用Processbuilder运行任何命令时,我们可以设置环境变量,即: 如果在此方法之前使用System.setProperties()设置了一些属性,则该属性可用于由ProcessBuilder启动的此过程吗? 问题答案: 获取 环境变量 。获取Java属性。环境变量是在OS级别指定的。通

  • 问题内容: 和以下代码之间有什么区别: 和 他们是同义词吗?在某些情况下,一个比另一个更可取吗?使用这两种方法时该做什么和不该做什么? 问题答案: Class.forName()将始终使用调用者的ClassLoader,而ClassLoader.loadClass()可以指定其他ClassLoader。我相信Class.forName也会初始化加载的类,而ClassLoader.loadClass