面试:Hibernate效率很低,如何优化?

缓存怎么弄的,语句怎么优化?

聊聊一级缓存,聊聊抓取策略和延迟加载,聊聊批量抓取

延迟加载:

一般不单独使用,和延迟加载一起使用

延迟加载:lazy(懒加载)

执行到该行代码的时候不会发送语句,真正使用这个对象的属性的时候才会发送sql语句进行查询。

  • 类级别延迟加载:指的是是通过load方法查询某个对象的时候是否采用延迟,通过class标签上的lazy来配置。

          • 让类级别延迟加载失效:1 lazy设为false 2 final修饰 3 调用Hibernate.initialize
  • 关联级别的延迟加载:指的是查询某个对象的关联对象的时候,是否采用延迟加载。session.get(Customer.class,1l) ,  customer.getLinkMans();
          • 抓取策略往往和关联级别的延迟加载一起使用,优化语句

类级别延迟加载配置:默认为true

关联级别的延迟加载配置:在映射文件被关联的对象上配置

抓取策略:

通过一个对象抓取到关联对象需要发送sql语句,sql语句如何发送,发送成什么格式通过策略进行配置

  • 通过<set> 或者发送一条子查询上通过fetch属性进行设置
  • fetch和这些标签上的lazy如何设置优化发送的语句

<set>上的fetch和lazy

fetch:抓取策略,控制sql语句格式

  • select:  默认值,发送普通select语句,查询关联对象,至少交互两次
  • join:   发送迫切左外连接查询关联,与数据库交互一次,发一条数据
  • subselect:发送一条子查询查询关联对象

lazy:延迟加载,控制查询关联对象的时候是否采用延迟

  • true:默认值,查询关联对象默认采用延迟加载
  • false:查询关联对象不采用延迟加载
  • extra :极其懒惰。

在实际开发中,一般都采用默认值。如果有特殊需求,可能配置join

各种配置的区别测试:

/**
* 在<set>上的fetch和lazy
*/
public class HibernateDemo4 {
@Test
/**
* 默认情况
*/
public void demo1() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
Customer customer = session.get(Customer.class, 1l);// 发送一条查询客户的sql
System.out.println(customer.getCust_name());
// 查看1号客户每个联系人的信息
for (LinkMan linkMan : customer.getLinkMans()) {// 发送一条根据客户查询联系人的sql
System.out.println(linkMan.getLkm_name());
}
transaction.commit();
} @Test
/**
* 设置fetch=select,lazy = true
*/
public void demo2() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
Customer customer = session.get(Customer.class, 1l);// 发送一条查询客户的sql
System.out.println(customer.getCust_name());
// 查看1号客户每个联系人的信息
for (LinkMan linkMan : customer.getLinkMans()) {// 发送一条根据客户查询联系人的sql
System.out.println(linkMan.getLkm_name());
}
transaction.commit();
} @Test
/**
* 设置fetch=select,lazy =false
*/
public void demo3() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
Customer customer = session.get(Customer.class, 1l);// 不延迟,直接发送两条语句,查询客户的sql,根据客户查询联系人的sql
System.out.println(customer.getCust_name());
System.out.println(customer.getLinkMans().size());// 发送一条select
// count(lkm_id)
// from transaction.commit();
} @Test
/**
* 设置fetch=select,lazy =extra
*/
public void demo4() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
Customer customer = session.get(Customer.class, 1l);// 发送一条查询客户语句
System.out.println(customer.getCust_name()); System.out.println(customer.getLinkMans().size());// 发送一条select
// count(lkm_id)
// from
transaction.commit();
} @Test
/**
* 设置fetch=join,lazy =失效
*/
public void demo5() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
Customer customer = session.get(Customer.class, 1l);// 发送一条迫切左外连接语句,两表全查
System.out.println(customer.getCust_name()); System.out.println(customer.getLinkMans().size());// 不发语句
transaction.commit();
} @Test
/**
* 设置fetch=subselect,lazy =true
*/
public void demo6() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
List<Customer> list = session.createQuery("from Customer").list();// 发送查询所有客户的sql语句
for (Customer customer : list) {
System.out.println(customer.getCust_name());
System.out.println(customer.getLinkMans().size());// 发送一条子查询
}
transaction.commit();
} @Test
/**
* 设置fetch=subselect,lazy =false
*/
public void demo7() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
// 查询1号客户
List<Customer> list = session.createQuery("from Customer").list();// 发送查询所有客户的sql语句,发送一条子查询
for (Customer customer : list) {
System.out.println(customer.getCust_name());
System.out.println(customer.getLinkMans().size());
}
transaction.commit();
} }

<many-to-one>上的fetch和lazy

fetch:抓取策略

  • select:默认值,发送普通sql查询关联对象
  • join:发送迫切左外连接

lazy:延迟加载,控制查询关联对象的时候是否采用延迟

  • proxy:默认值,具体的取值和另一端<class>上lazy的值相同
  • false:查询关联对象,不采用延迟
  • no-proxy(不会使用)

一般使用默认值

各种配置的区别测试:

/**
* 在<many-to-one>上的fetch和lazy
*/
public class HibernateDemo5 {
@Test
/**
* 默认情况
*/
public void demo1() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
LinkMan linkMan = session.get(LinkMan.class, 1l);// 发送一条查询联系人的语句
System.out.println(linkMan.getLkm_name());
System.out.println(linkMan.getCustomer());// 发送一条select查询联系人所关联的客户的语句
transaction.commit();
} @Test
/**
* fetch = "select" lazy = "proxy",默认值
*/
public void demo2() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
LinkMan linkMan = session.get(LinkMan.class, 1l);// 发送一条查询联系人的语句
System.out.println(linkMan.getLkm_name());
System.out.println(linkMan.getCustomer());// 发送一条select查询联系人所关联的客户的语句
transaction.commit();
} @Test
/**
* fetch = "select" lazy = "false"
*/
public void demo3() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
LinkMan linkMan = session.get(LinkMan.class, 1l);// 不延迟,发送一条查询联系人的语句,发送一条select查询联系人所关联的客户的语句
System.out.println(linkMan.getLkm_name());
System.out.println(linkMan.getCustomer());
transaction.commit();
} @Test
/**
* fetch = "join" lazy = 失效
*/
public void demo4() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
LinkMan linkMan = session.get(LinkMan.class, 1l);// 发送一条迫切左外连接,两表全查
System.out.println(linkMan.getLkm_name());
System.out.println(linkMan.getCustomer());
transaction.commit();
} }

批量抓取:

批量抓取:一批关联对象一起抓取,batch-size,

/**
* Hibernate批量抓取
*/
public class HibernateDemo6 {
@Test
/**
* 获取客户的时候批量抓取联系人,在Customer.hbm.xml中设置set标签batch-size="4"
*/
public void demo1() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
List<Customer> list = session.createQuery("from Customer").list();
for (Customer customer : list) {
System.out.println(customer.getCust_name());
for (LinkMan linkMan : customer.getLinkMans()) {
System.out.println(linkMan.getLkm_name());
}
}
transaction.commit();
} @Test
/**
* 获取联系的时候批量抓取客户,在Customer.hbm.xml中设置class标签batch-size="4"
*/
public void demo2() {
Session session = HibernateUtils.getCurrentSession();
Transaction transaction = session.beginTransaction();
List<LinkMan> list = session.createQuery("from LinkMan").list();
for (LinkMan linkMan : list) {
System.out.println(linkMan.getLkm_name());
System.out.println(linkMan.getCustomer().getCust_name());
}
transaction.commit();
}
}

八 Hibernate延迟加载&抓取策略(优化)的更多相关文章

  1. Hibernate框架笔记04HQL_QBC查询详解_抓取策略优化机制

    目录 1. Hibernate的查询方式 1.1 方式一:OID查询 1.2 方式二:对象导航查询 1.3 方式三:HQL方式 1.4 方式四:QBC查询 1.5 方式五:SQL查询 2. 环境搭建 ...

  2. Hibernate的抓取策略(优化)

    延迟加载的概述 什么是延迟加载 延迟加载:lazy(懒加载).执行到该行代码的时候,不会发送语句去进行查询,在真正使用这个对象的属性的时候才会发送SQL语句进行查询. 延迟加载的分类 l  类级别的延 ...

  3. Hibernate之抓取策略

    时间:2017-1-23 19:08 --区分延迟和立即检索1.立即检索    当执行某行代码时,会马上发出SQL语句进行查询.    例如:get()2.延迟检索    当执行某行代码时,不会马上发 ...

  4. Hibernate的抓取策略

    立即检索:当执行某行代码的时候,马上发出SQL语句进行查询(get())延迟检索:当执行某行代码的时候,不会马上发出SQL语句进行查询.当真正使用这个对象的时候才会发送SQL语句(load()) 类级 ...

  5. Hibernate学习第4天--HQL——QBC查询详解,抓取策略优化。

    上次课回顾: l  Hibernate的一对多 n  表与表之间关系 u  一对多关系 u  多对多关系 u  一对一关系 n  Hibernate的一对多配置 u  搭建Hibernate基本环境 ...

  6. 【Hibernate】抓取策略

    一.区分延迟和立即检索 二.类级别检索和关联级别检索 一.区分延迟和立即检索 立即检索: 当执行某行代码的时候,马上发出SQL语句进行查询. get() 延迟检索: 当执行某行代码的时候,不会马上发出 ...

  7. 【Hibernate 8】Hibernate的调优方法:抓取策略

    在上一篇博客中,介绍了Hibernate的缓存机制.合理的配置缓存,可以极大程度上优化Hibernate的性能.这篇博客,介绍另外一个调优方式:抓取策略. 一.什么是抓取策略 抓取策略(fetchin ...

  8. Hibernate(十四)抓取策略

    抓取策略: 抓取策略是当应用程序需要在(Hibernate实体对象图的)关联关系间进行导航的时候,Hibernate如何获取关联对象的策略.Hibernate的抓取策略是Hibernate提升性能的一 ...

  9. Hibernate 抓取策略

    抓取策略: 为了改变SQL语句执行的方式 当应用程序需要在Hibernate实体对象的关联关系间进行导航的时候,Hibernate如何获取关联对象的策略 抓取策略可以在O/R映射的元数据中声明,也可以 ...

随机推荐

  1. IDEA & MAVEN配置代理(没用)

    1. IDEA配置代理: 2. maven配置代理: 在maven中配置代理,主要配置编辑~/.m2/settings.xml文件的<proxies> socks5类型: <id&g ...

  2. MySQL复制方法

    MySQL的二进制日志,MySQL复制原理,MySQL主从模式搭建,MySQL双主模式搭建,MySQL级联模式搭建,MySQL半同步模式复制 一.二进制日志 1.概念 MySQL的二进制日志(bina ...

  3. 解决springmvc拦截器拦截静态资源的两种方式

    1.是采用<mvc:default-servlet-handler />,(一般Web应用服务器默认的Servlet名称是"default",所以这里我们激活Tomca ...

  4. cmd设置utf8编码

    在中文windows系统中,如果一个文本文件是utf-8编码的,那么在cmd.exe命令行窗口(所谓的dos窗口)中不能正确显示文件中的内容.在默认情况下,命令行窗口中使用的代码页是中文或者美国的,即 ...

  5. 操作系统OS - 反置页表

    1. https://blog.csdn.net/wuyuegb2312/article/details/16359821 2. https://www.youtube.com/watch?v=YQ3 ...

  6. C++的const成员函数

    我们知道,在C++中,若一个变量声明为const类型,则试图修改该变量的值的操作都被视编译错误.例如, const char blank = ‘’; blank = ‘\n’;  // 错误 面向对象 ...

  7. java 并发线程锁

     1.同步和异步的区别和联系 异步,执行完函数或方法后,不必阻塞性地等待返回值或消息,只需要向系统委托一个异步过程,那么当系统接收到返回 值或消息时,系统会自动触发委托的异步过程,从而完成一个完整的流 ...

  8. Java基础 -3.3

    位运算 指的是可以直接进行二进制数据的计算处理,主要是:与(&).或(|).非(!).异或(^).反码(~).移位处理(>>.<<) 与运算 public class ...

  9. Python流程控制-2 条件判断

    条件判断 条件判断是通过一条或多条判断语句的执行结果(True或者False)来决定执行的代码块. 在Python语法中,使用if.elif和else三个关键字来进行条件判断. if语句的一般形式如下 ...

  10. sshd免密登陆

    用途:默认情况下,当A主机(1.1.1.1)远程通过ssh命令登陆到B主机(2.2.2.2)上,需要输入B主机的密码.免密登陆的效果为,A通过ssh命令登录到B时,不需要输入密码就可以登录,便于管理. ...