zoukankan      html  css  js  c++  java
  • 第七章 Hibernate性能优化

    一对一关联

    实体类关系

    一对多

    多对多

    一对一

    Hibernate提供了两种映射一对一关联关系的方式:按照外键映射按照主键映射。下面以员工账号和员工档案表为例,介绍这两种映射方式,并使用这两种映射方式分别完成以下持久化操作:

    (1)保存员工档案的同时分配给员工一个账号。

    (2)加载员工档案的同时加载账号信息。

    1.按照外键映射

    步骤一:创建实体类Users1Resume1

    Users1创建如下:
    
    public class Users1 {
    
    private Integer userid;
    
    private String username;
    
    private String userpass;
    
    private Resume1 resume1;
    
    }
    
    Resume1创建如下:
    
    public class Resume1 {
    
    private Integer resid;
    
    private String resname;
    
    private String rescardno;
    
    private Users1 users1;
    
    }

    步骤二:配置文件Users1.hbm.xmlResume1.hbm.xml

    Users1.hbm.xml配置如下:

    <?xml version="1.0" encoding="UTF-8"?>
    
    <!DOCTYPE hibernate-mapping PUBLIC 
    
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    
    <hibernate-mapping package="cn.happy.onetoone_fk">
    
       <class name="Users1" table="USERS1">
    
         <id name="userid" column="USERID" >
    
            <generator class="native"></generator>
    
         </id>
    
         <property name="username" column="USERNAME" type="string"></property>
    
         <property name="userpass" column="USERPASS" type="string"></property>
    
         <one-to-one name="resume1" class="Resume1" property-ref="users1"></one-to-one>
    
       </class>
    
    </hibernate-mapping>
    
    Resume1.hbm.xml配置如下:
    
    <?xml version="1.0" encoding="UTF-8"?>
    
    <!DOCTYPE hibernate-mapping PUBLIC 
    
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    
    <hibernate-mapping package="cn.happy.onetoone_fk">
    
       <class name="Resume1" table="RESUME1">
    
         <id name="resid" column="RESID" >
    
            <generator class="native"></generator>
    
         </id>
    
         <property name="resname" column="RESNAME" type="string"></property>
    
         <property name="rescardno" column="RESCARDNO" type="string"></property>
    
         <many-to-one name="users1" class="Users1" cascade="all" column="RESUSERID" unique="true"></many-to-one>
    
       </class>
    
    </hibernate-mapping>
    
    步骤三:测试方法书写
    
    @Test
    
      public void testAdd(){
    
       Session session = HibernateUtil.getSession();
    
       Transaction tx=session.beginTransaction();
    
       //创建一个用户对象
    
       Users1 u1=new Users1("happy","1");
    
       //创建一个档案对象
    
       Resume1 r1=new Resume1("小学档案","happy01");
    
       u1.setResume1(r1);
    
       r1.setUsers1(u1);
    
       //保存r1自动保存u1
    
       session.save(r1);
    
       tx.commit();
    
       System.out.println("ok===");
    
      }

    1.按照主键映射 

    步骤一:创建实体类Users2和Resume2
    Users2的创建如下:
    public class Users2 {
        private Integer userid;
        private String username;
        private String userpass;
        private Resume2 resume2;
    } 
    Resume2的创建如下:
    public class Resume2 {
        private Integer resid;
        private String resname;
        private String rescardno;
        private Users2 users2;
    }
    步骤二:配置文件Users1.hbm.xml和Resume1.hbm.xml
    Users1.hbm.xml配置如下:
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE hibernate-mapping PUBLIC 
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    <hibernate-mapping package="cn.happy.onetoone_pk">
       <class name="Users2" table="USERS2">
         <id name="userid" column="USERID" >
            <generator class="foreign">
              <param name="property">resume2</param>
            </generator>
         </id>
         <property name="username" column="USERNAME" type="string"></property>
         <property name="userpass" column="USERPASS" type="string"></property>
         <one-to-one name="resume2" class="Resume2" constrained="true"></one-to-one>
       </class>
    </hibernate-mapping>
    Resume2.hbm.xml配置
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
                                "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    <hibernate-mapping package="cn.happy.onetoone_pk">
     <class name="Resume2" table="RESUME2">
      <id column="RESID" name="resid">
       <generator class="native"/>
      </id>
      <property column="RESNAME" name="resname" type="string"/>
      <property column="RESCARDNO" name="rescardno" type="string"/>
      <one-to-one  name="users2" cascade="all" class="Users2"/>
     </class>
    </hibernate-mapping>

    组件映射

      建立关系数据模型的一个重要原则是在不会导致数据冗余的前提下,尽可能减少数据库表的数目及表之间的外键参照关系。以员工信息为例,员工信息中有员工的家庭地址信息,如果把地址信息单独放在一张表中,然后建立员工信息表和地址信息表之间的外键参照关系,当每次查询员工信息时,都需建立者两个表的连接。建立表的连接是很耗时的操作,为了提高数据库运行性能,可以把这两张表的信息整合在一张员工信息表EMPINFO

    步骤一:创建EmpHomeAddress和EmpInfo
    public class EmpHomeAddress {
        private String ehomestreet;
        private String ehomecity;
        private String ehomeprovince;
        private String ehomezipcode;
        private EmpInfo empinfo;
    }
    EmpInfo创建如下:
    public class EmpInfo {
        private Integer eid;
        private String ename;
        private EmpHomeAddress ehome;
    }
    步骤二:创建配置文件EmpInfo.hbm.xml
    <?xml version="1.0" encoding="UTF-8"?>
    <!DOCTYPE hibernate-mapping PUBLIC 
        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
        "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
    <hibernate-mapping package="cn.happy.component">
       <class name="EmpInfo" table="EMPINFO">
           <id name="eid" column="EID">
            <generator class="native"></generator>
           </id>
           <property name="ename" column="ENAME" type="string"></property>
           <component name="ehome" class="EmpHomeAddress">
              <parent name="empinfo"/>
              <property name="ehomestreet" column="EHOMESTREET" type="string"></property>
              <property name="ehomecity" column="EHOMECITY" type="string"></property>
              <property name="ehomeprovince" column="EHOMEPROVINCE" type="string"></property>
              <property name="ehomezipcode" column="EHOMEZIPCODE" type="string"></property>
           </component>
       </class> 
    </hibernate-mapping>
    步骤三:书写测试类
     @Test
          public void testAdd(){
               Session session = HibernateUtil.getSession();
               Transaction tx=session.beginTransaction();
               //创建一个员工对象
               EmpInfo emp=new EmpInfo();
               emp.setEname("张靓颖");
               
               //创建一个员工地址对象
               EmpHomeAddress address=new EmpHomeAddress();
               address.setEhomecity("北京");
               address.setEhomeprovince("北京");
               address.setEhomestreet("五道口");
               address.setEhomezipcode("100000");
               address.setEmpinfo(emp);
               emp.setEhome(address);
               session.save(emp);
               tx.commit();
               System.out.println("ok===");
          }

    Hibernate缓存

    缓存:是计算机领域的概念,它介于应用程序和永久性数据存储源之间。

    缓存:一般人的理解是在内存中的一块空间,可以将二级缓存配置到硬盘。用白话来说,就是一个存储数据的容器。我们关注的是,哪些数据需要被放入二级缓存。

    作用:降低应用程序直接读写数据库的频率,从而提高程序的运行性能。缓存中的数据是数据存储源中数据的拷贝。缓存的物理介质通常是内存。

    04.一级缓存

    01.Session内的缓存即一级缓存,内置且不能被卸载,一个事务内有效。在这个空间存放了相互关联的Java对象,这种位于Session缓存内的对象也别称为持久化对象,Session负责根据持久化对象的状态变化来同步更新数据库。

    02.Session为应用程序提供了管理缓存的方法:

    evict(Object o)

    clear()

    03.金牌结论一级缓存

         一级缓存的生命周期session的生命周期一致,当前session一旦关闭,一级缓存就消失了,因此一级缓存也叫session级的缓存或事务级缓存,一级缓存只存实体对象,它不会缓存一般的对象属性(查询缓存可以),即当获得对象后,就将该对象缓存起来,如果在同一session中再去获取这个对象时,它会先判断在缓存中有没有该对象的id,如果有则直接从缓存中获取此对象,反之才去数据库中取,取的同时再将此对象作为一级缓存处理。

    以下方法支持一级缓存:金牌结论

    * get() 
        * load() 
        * iterate(查询实体对象) 
    其中 Query Criterialist() 只会缓存,但不会使用缓存(除非结合查询缓存)。

    二级缓存(面试题)

    开发中的用途没有面试带来作用大。

    二级缓存是进程(N个事务)或集群范围内的缓存,可以被所有的Session共享,在多个事务之间共享

    二级缓存是可配置的插件

    二级缓存的配置使用(ehcache缓存)

       *1.引入如下jar包。

          ehcache-1.2.3.jar  核心库

          backport-util-concurrent.jar  

          commons-logging.jar

       *2.配置Hibernate.cfg.xml开启二级缓存

    <property name="hibernate.cache.use_second_level_cache">true</property>

      *3.配置二级缓存的供应商

    <property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>

      *4.指定使用二级缓存的类

           方案一:在*.hbm.xml中配置

           <class>元素的子元素下添加chche子节点,但该配置仅会缓存对象的简单属性,若希望缓存集合属性中的元素,必须在set元素中添加<cache>子元素

          <class name="Student" table="STUDENT">

        <cache usage="read-write"/>

    方案二:在大配置文件(hibernate.cfg.xml)中配置

    位置有限定

    Multiple annotations found at this line:

    - The content of element type "session-factory" must match "(property*,mapping*,(class-cache|collection-

     cache)*,event*,listener*)".

    - Start tag of element <session-factory>

         <class-cache usage="read-write" class="cn.happy.entity.Student"/>

    <collection-cache usage="read-write" collection=""/>

    *5.src下添加ehcache.xml文件,从etc获取文件即可。

    解析 :出现如下错误因为没有添加二级缓存所需jar

    org.hibernate.HibernateException: could not instantiate RegionFactory [org.hibernate.cache.impl.bridge.RegionFactoryCacheProviderBridge]

    结果

    二级缓存原理

  • 相关阅读:
    PHP获取某年第几周的开始日期和结束日期
    Java多线程(一)
    Java多线程学习
    使用反射实现 webdriver page 类
    PageObjects 设计模式
    Selenium WebDriver 工作原理
    Selenium2.0 Webdriver 随笔
    Selenium-Grid2 配置RemoteWebDriver
    Java多线程基础(二)
    Java多线程基础(一)
  • 原文地址:https://www.cnblogs.com/LJN520/p/5842983.html
Copyright © 2011-2022 走看看