hibernate之关于1+N的问题

2023-11-16


【Hibernate】之关于1+N的问题


    1+N问题,人叫做N+1问题,至今未统一,在这里我会告诉大家我为什么称之为1+N问题!

    

    什么情况下会产生1+N问题;


   在实际的项目开发中,我们配置的一对多,或者是多对一,在查询的时候会产生一种现象。


   例如,人(Person)和组(Group)


当我们在查询(多的一方)Hibernate会直接发SQL把相关的(一的一方)也查询出来;


当然,这种情况问题说大也不是特别大,但是,当我们数据量较大,数据库的性能就是不得不考虑的事情,ok?


首先我们得弄明白为什么我list() Person的时候,会发查询Group的语句?我们知道在many-to-oneFetchType默认是EAGEREAGER的意思可以理解为急加载,加载一个实体时,定义急加载的属性会立即从数据库中加载。所以Hibenrate加载Person属性值的时候发现Person的有一个字段属性与Group有关联,那么,Hibernate默认就立刻发请求将关联对象取出。


但是相反,我们list Group的时候,我们发现就不会出现这种现象,这是因为one-to-many默认的FetchTyp默认是LAZY,懒加载的意思,加载一个实体时,定义懒加载的属性不会马上从数据库中加载。只有我们需要的时候,比如group.getPersons().size() Hibernate检测你需要之后,才会发SQL请求!


ok,上面说了1+N问题出现的原理,那下面用程序来证明:

@Entity
@Table(name="t_group")
publicclass Group {
    private Integer id;
    private String name;
    private Set<Person> persons=newHashSet<Person>();
     
    @OneToMany
    public Set<Person> getPersons() {
       returnpersons;
    }
    publicvoid setPersons(Set<Person> persons) {
       this.persons = persons;
    }
    @Id
    @GeneratedValue
    public Integer getId() {
       returnid;
    }
    publicvoid setId(Integer id) {
       this.id = id;
    }
    @Column(name="g_name")
    public String getName() {
       returnname;
    }
    publicvoid setName(String name) {
       this.name = name;
    }
}

@Entity
@Table(name="p_person")
publicclass Person {
    private Integer id;
    private String name;
    private Integer age;
    private Group group;
     
    @ManyToOne
    @JoinColumn(name="group_id")
    public Group getGroup() {
       returngroup;
    }
    publicvoid setGroup(Group group) {
       this.group = group;
    }
    @Id
    @GeneratedValue
    public Integer getId() {
       returnid;
    }
    publicvoid setId(Integer id) {
       this.id = id;
    }
    @Column(name="p_name")
    public String getName() {
       returnname;
    }
    publicvoid setName(String name) {
       this.name = name;
    }
    @Column(name="p_age")
    public Integer getAge() {
       returnage;
    }
    publicvoid setAge(Integer age) {
       this.age = age;
    }
}

现在我们查询Person,看它发送的SQL语句


//我们发现,在查询Person的时候,发送的SQL语句
    @Test
    publicvoid findTest1(){
       Session s=sessionFactory.getCurrentSession();
       s.beginTransaction();
       List<Person> persons=s.createQuery("from Person").list();
       for(Person person:persons){
           System.out.println(person.getName()+"----"+person.getId());    }
       s.getTransaction().commit();
    }

查看SQL语句

(除了第一条是查询Person的语句,后面10条都是查询Group的语句)

11:02:13,036 DEBUGSQL:111 - 
    select
        person0_.id as id1_,
        person0_.p_age as p2_1_,
        person0_.group_id as group4_1_,
        person0_.p_name as p3_1_ 
    from
        p_person person0_
11:02:13,078 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,096 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,100 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,102 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,103 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,105 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,107 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,110 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,113 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
11:02:13,115 DEBUGSQL:111 - 
    select
        group0_.id as id0_0_,
        group0_.g_name as g2_0_0_ 
    from
        t_group group0_ 
    where
        group0_.id=?
张三0----1
张三1----2
张三2----3
张三3----4
张三4----5
张三5----6
张三6----7
张三7----8
张三8----9
张三9----10

这里我只需要查询的是Person的信息,只需要一条SQL语句就够了,但是现在多发了10条查询Group的语句,所以在这里我称之1+N问题。OK?

怎么解决Hibernate的1+N问题?

前面提到FetchType对,没错。

第一种方法,就是将many-to-one的FetchType值设为 LAZY  意思是告诉Hibernate  我需要的时候,你再发SQL请求。

@ManyToOne(fetch=FetchType.LAZY)

第二种方法,采用createCriteria这种事默认采用表连接的形式,也是可以解决。

第三种方法,left join fetch 做个表连接 也是可以的。

第四种方法,在一的一方加上@BatchSize  指定一个值(指定的值代表就是in()里面的值,发的SQL)也是可以的,但是不建议这么做,毕竟@BatchSize真正解决不了1+N这种问题,最多只能是少发几条SQL而已,大家有兴趣可以慢慢研究!

@Test
    publicvoid findTest1(){
       Session s=sessionFactory.getCurrentSession();
       s.beginTransaction();
//     List<Person>persons=(List<Person>)s.createCriteria(Person.class).list();//解决方法1
       List<Person> persons=s.createCriteria(Person.class).list();
    //  List<Person>persons=s.createQuery("from Person p left join fetch p.groupg").list();//解决方法3
       for(Person person:persons){
           System.out.println(person.getName()+"----"+person.getId());
//         System.out.println(person.getGroup().getName());
       }
       s.getTransaction().commit();
    }

ok,1+N问题,是Hibernate面试最常考的题,同时也是性能优化一种常见的手段。本文详细讲解了1+N问题的原理和解决办法!有问题举手!









本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

hibernate之关于1+N的问题 的相关文章

随机推荐

  • redis基础4——RDB持久化、AOF持久化全面深入解读

    文章目录 一 redis持久化机制 1 1 持久化的背景 1 2 两种持久化概念 1 2 1 快照方式 RDB 1 2 2 文件追加方式 AOF 1 3 rdb持久化 Redis Database 1 3 1 快照原理 1 3 2 触发机制
  • 组合聚合的概念

    聚合的概念 聚合 Aggregation 关系是关联关系的一种 是强的关联关系 聚合是整体和个体之间的关系 例如 汽车类与引擎类 轮胎类 以及其它的零件类之间的关系便整体和个体的关系 聚合关系也是通过实例变量实现的 在聚合关系中 两个类是处
  • shell脚本中遇到错误时中断程序运行,不再执行后面的程序

    shell脚本中遇到错误时中断程序运行 不再执行后面的程序 当你在脚本中写了一连串的代码时 如果后面的代码需要前面代码执行正确才能继续执行时 你可以使用set e vim test sh新建一个脚本文件 bin bash 设置程序出错时不再
  • 【软件工程】静态测试与动态测试

    静态测试 桌前检查 代码走查 代码审查 动态测试 黑盒测试 等价类划分 确定无效与有效等价类 设计用例尽可能多的覆盖有效类 设计用例只覆盖一个无效类 边界值分析 处理边界情况时最容易出错 选取的测试数据应该恰等于 稍小于或稍大于边界值 错误
  • python爬虫返回百度安全验证

    我一开始用的是requests库 header加了accept和user agent 这是一开始的代码 import requests headers Accept text html application xhtml xml appli
  • SpringBoot项目使用EasyPoi实现导入导出,就是这么的丝滑

    在项目的开发工程中 经常有导入导出数据的常见功能场景 Apache的POI是处理导入导出中最常用的 但是其原生的用法太复杂 很繁琐 总是在Copy 无意间发现一款简单粗暴的神器EasyPoi EasyPoi也是基于POI的 在SpringB
  • 使用vpd进行行级控制

    在系统用户下 1 创建vpd用户 create user vpd identified by 123456 grant resource connect to vpd grant execute on dbms rls to vpd gra
  • 高德地图, 动态绘制多个marker 并 随着地图缩放, 判定marker之间的距离, 显示不同 marker 效果

    转载
  • JVM系统线程

    虚拟机线程 这种线程的操作时需要JVM达到安全点才会出现 这些操作必须在不同的线程中发生的原因是他们都需要JVM达到安全点 这样堆才不会变化 这种线程的执行类型包括 stop the world 的垃圾收集 线程栈收集 线程挂起以及偏向撤销
  • MFC Windows程序设计1_3

    使用VS2008生成MFC程序 选择对话框形式 主要的需要注意的 在App类中 重写InitInstance 函数 MyDlg dlg m pWindow dlg dlg doModal return FALSE 注意InitInstanc
  • 读书有感:《失业的程序员》

    失业的程序员 是我在三天前心血来潮找来的一本书 这是一本极其易读 风趣横生的关于程序员从失业到创业的小说类书籍 书中主人公从一开始辞职失业 到整合资源开始创业 再到最后看似创业已经稳定却是艰难险阻 创业团队也从一开始的 2 人 到 10 多
  • HTML5(十一)——WebSocket 基础教程

    一 为什么要学 WebSocket websocket 是 HTML5 提供的一种长链接双向通讯协议 使得客户端和服务器之间的数据交换更简单 允许服务端主动向客户端推送数据 并且客户端与服务端只需连接一次 就可以保持长久连接 并进行数据通信
  • Unity 委托 (Delegate) 的简单理解以及实现

    委托相当于把某一个方法当成参数 当执行委托的时候就相当于执行了方法 所以这个方法必须和委托具有相同的参数类型 委托的简单实现 using UnityEngine 委托 代理 是存有对某个方法的引用的一种引用类型变量 委托语法 delegat
  • 蓝桥杯冲击01 - 质数篇

    目录 前言 一 质数是什么 二 易错点 三 试除法判断是否为质数 四 质数常考三大模型 五 真题练手 前言 距离蓝桥杯还有一个月 高效复习蓝桥杯知识 质数相关的题目在蓝桥杯中经常出现 例如 2016年蓝桥杯省赛初赛第四题就是要求判断一个数是
  • 基于宽表的数据建模

    一 业务背景 1 1 数据建模现状 互联网企业往往存在多个产品线 每天源源不断产出大量数据 这些数据服务于数据分析师 业务上的产品经理 运营 数据开发人员等各角色 为了满足这些角色的各种需求 业界传统数仓常采用的是经典分层模型的数仓架构 从
  • 部分安卓端ncnn模型推理输出数据存在大量-nan和nan的问题

    原文issue链接 部分安卓端ncnn模型推理输出数据存在大量 nan的问题 Issue 3607 Tencent ncnn github com 问题描述 onnx ncnn模型在pc端推理输出结果正确且基本一致 在部分安卓设备上使用同一
  • 剑指 Offer 41. 数据流中的中位数(java+python)

    如何得到一个数据流中的中位数 如果从数据流中读出奇数个数值 那么中位数就是所有数值排序之后位于中间的数值 如果从数据流中读出偶数个数值 那么中位数就是所有数值排序之后中间两个数的平均值 例如 2 3 4 的中位数是 3 2 3 的中位数是
  • 微信小程序内添加腾讯地图 导航

    1 微信小程序 设置 第三方设置 插件管理 添加插件 如果搜索不到腾讯位置服务路线规划插件 可搜索如下地址直接添加腾讯位置服务路线规划 小程序插件 微信公众平台 问题 提示添加失败 项目主体不符的 可以通过 服务 微信服务市场 搜索 腾讯位
  • Python3 sorted自定义排序

    Python3的sorted函数具有key参数 可以进行自定义的排序 但并不是直接将自定义比较函数赋给key 这一点和C 是不一样的 在python3中 需要import functools来传递比较函数 注意 在Python3中 自定义比
  • hibernate之关于1+N的问题

    Hibernate 之关于1 N的问题 1 N问题 也有人叫做N 1问题 至今未统一 在这里我会告诉大家我为什么称之为1 N问题 什么情况下会产生1 N问题 在实际的项目开发中 我们配置的一对多 或者是多对一 在查询的时候会产生一种现象 例