随笔-124  评论-49  文章-56  trackbacks-0
 
     摘要: 在hql中关键字不区分大小写,但是属性和类名区分大不写 简单属性查询[重要] 1  单一属性查询,返回结果集属性列表,元素类型和实体类中相应的属性类型一致  List students=session.createQuery("select name from Student").list();   &nb...  阅读全文
posted @ 2009-11-03 17:14 junly 阅读(493) | 评论 (0)编辑 收藏
一级缓存

  * 一级缓存是缓存实体对象的
  * 如果管理一级缓存
    一级缓存无法取消,但可以管理clear(),evict()
  * 一级缓存和session的生命周期一致,一级缓存也叫session级的缓存或事务级缓存
  * 如何避免一次性大量的实体数据入库导至内存溢出
    先flush,再clear
  * 如何管理一级缓存
    load,get,iterate,save都支持一级缓存
如果数据量特别大,考虑采用jdbc实现,如查jdbc也不能满足要求可以考虑采用数据本身的特定导入工具
Student student=(Student)session.load(Student.class,1);
System.out.println(
"studnet.name="+student.getName());
//不会发出sql,因为load使用缓存
Student student=(Student)session.load(Student.class,1);
System.out.println(
"studnet.name="+student.getName());

二级缓存

  * 二级缓存是缓存实体对象的,普通属性不会缓存
  * 二级缓存是进程级的缓存,也称为SessionFactory级的缓存,可以被所有的session共享
    二级缓存的生命周期和SessionFactory是一致的,可以用SessionFactory管理二级缓存
  * 二级缓存的配置和使用
    1 加入ehcache的jar包;
    2 拷贝ehcache.xml文件到src目录下;
    3 开启二级缓存,默认是打开的。配置hibernate.cfg.xml
    hibernate.cfg.xml文件
    开启二级缓存
<property name="hibernate.cache.use_second_level_cache">true</property>
    指定缓存产品提供商
<property name="hibernate.cache.provider_class">org.hibernate.cache.EhCacheProvider</property>
    4 指定那些对象使用二级缓存(两种方运河)
      * 在映射文件中采用<cache>标签
 
<class name="com.my.hibernate.User" table="t_user">
        
<cache usage="read-only"/>
     
<id name="id">
      
<generator class="native"/>
     
</id>
     
<property name="name"/>
     
<many-to-one name="group" column="groupid" cascade="save-update"/>
    
</class>
    * 在hibernate.cfg.xml文件中,采用<class-cache/>标签
<class-cache class="com.bjsxt.hibernate.Studnet" usage="read-only"/>

管理二级缓存

factory.evict(Student.class);
factory.evict(Student.class,1);

一级缓存和二级缓存的交互问题
   1 不设置(默认)会写入二级缓存,也会读出
   2 GET只读而不写入二级缓存

   session.setCacheMode(CacheMode.GET);
   Student student
=(Student)session.load(Student.class,1);

   3 PUT只写入二级缓存页不读取

   session.setCacheMode(CacheMode.PUT);
   Student student
=(Student)session.load(Student.class,1);


查询缓存

查询缓存是针对普通属性结果集的缓存
对实体对象的结果只缓存id

查询缓存的生命周期,当前关联的表发生修改,那么查询缓存生命周期结束

查询缓存的配置和使用
1 起用查询缓存
  * 配置hibernate.cfg.xml文件

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

  * 在程序中显式起用

query.setCacheable(true);

2 Session和查询缓存生命周期没有关系
3 查询缓存对query.iterate()不起作用,只用对query.list()起作用

posted @ 2009-11-03 16:49 junly 阅读(326) | 评论 (0)编辑 收藏

悲观锁

悲观锁的实现,通常依赖于数据库机制,在整个过程中将数据锁定,其它任何用户都不能读取或修改
session.load(Inventory.class, 1, LockMode.UPGRADE);

乐观锁

大多数基于数据版本记录机制(version)实现,一般是在数据库表中加入一个version字段
读取数据时将版本号一同读出,之后更新数据时版本号加一,如果提交数据时片本号小于
或等于数据库表中的版本号,则认为数据是过期的,否则给予更新。

1 Inventory.java:
 
private int id;
private String name;
private int count;
private int version;
//version版本号由数据库维护,我们不用管
2 Inventory.hbm.xml
<class name="Inventory" table="t_inventory2" optimistic-lock="version">
        
<id name="id">
            
<generator class="native"/>
        
</id>
        
<property name="name"/>
        
<property name="count"/>
        
<property name="version"/>
    
</class>
posted @ 2009-11-03 16:40 junly 阅读(207) | 评论 (0)编辑 收藏

session flush测试:

session flush方法主要做了两件事:
1 清理缓存
2 执行sql(不是提交事务)

session在什么情况下执行flush
1 默认在事务提交时
2 显式的调用flush
3 在执行查询前,如:iterate

hibernate按照save(insert),update,delete顺序提交相关的操作
------------------------------------------------------------------------

  <id name="id">
   <generator class="uuid"/>
  </id>
  因为id的主键生成策略采用的是uuid,所以调用完save后,只是将user对象纳入到session的管理
  不会发出insert语句,但是id已经生成,session中existsInDatebase状态为false
  session.save(user);
  调用flush,hibernate会清理缓存,执行sql
  如果数据库的隔离级别为提交读,那么我们可以看到flush过的数据
  并且session中existsInDatebase状态变为true
  session.flush();
  默认情况下commit操作会先执行者flush清理缓存,所以不用显式的调用flush
  commit后数据无法回滚
  session.getTransaction().commit();
  
  <id name="id">
   <generator class="native"/>
  </id>
  如果id的主键生成策略采用的是native,调用save(user)时会发出insert语句,返回由数据库生成的id,
  user对象纳入到session的管理,session中existsInDatebase状态为true
-----------------------------------------------------------------
  <id name="id">
   <generator class="uuid"/>
  </id>
  session.save(user);
  将user对象从session中逐出,即session的EntiryEntries属性中逐出
  session.evict(user);//清理缓存
  无法成功提交,因为hibernate在清理缓存时,在session的insertions集合中取出user对象进行insert操作后
  需要更新entityEntries属性中的existsnDatabase为true,而我们采用evict已经将user从session中
  逐出了,所以找不到相关数据,无法更新,抛出异常
  session.getTransaction().commit();
----------------------------------------------------------------
  <id name="id">
   <generator class="uuid"/>
  </id>
  session.save(user);
  flush后hibernate会清理缓存,会将user对象保存到数据库中,将session中的insertions中的user
  清除,并且设置session中existsInDatabase的状态为true
  session.flush(user);
  将user对象从session中逐出,即session的EntityEntries属性中逐出
  session.evict(user);//清理缓存
  可以成功提交,因为hibernate在清理缓存时,在session的insertions集合中无法找到user对象
  所以就不会发出insert语句,也不会更新session中的existsInDatabase的状态
  session.getTransaction().commit();
-----------------------------------------------------------------
  <id name="id">
   <generator class="native"/>
  </id>
  session.save(user);
  将user对象从session中逐出,即session的EntityEntries属性中逐出
  session.evict(user);//清理缓存
  可以成功提交,因为hibernate在清理缓存时,在session的insertions集合中无法找到user对象
  所以就不会发出insert语句,也不会更新session中的existsInDatabase的状态
  session.getTransaction().commit();
-----------------------------------------------------------------
    <id name="id">
   <generator class="assigned"/>
  </id>
  session.save(user);
  user.setName("张三");
  session.update(user);
  User user2=new User();
  user2.setId("003");
  user2.setName("李四");
  session.getTransaction().commit();
  结果:
  insert into ...
  insert into ...
  update t_user ...
  hibernate按照save(insert),update,delete顺序提交相关的操作
-----------------------------------------------------------------
   <id name="id">
   <generator class="assigned"/>
  </id>
  session.save(user);
  user.setName("张三");
  session.update(user);
  因为我们在session.update(user)后执行了flush,所以在以commit清理缓存时执行flush前的sql就不会发出.
  session.flush();//在这里flush操作就可以了
  
  User user2=new User();
  user2.setId("003");
  user2.setName("李四");
  session.getTransaction().commit();
  结果:
  insert into ...
  update t_user ...
  insert into ...
  按照我们想要顺序save(insert),update,save(insert)的顺序提交操作
-----------------------------------------------------------------

posted @ 2009-11-03 16:37 junly 阅读(1216) | 评论 (1)编辑 收藏

抓取策略(单端代理的批量抓取)
1 保持默认,也就是fetch="select"
  <many-to-one name="classes" column="classesid" cascade="save-update"/>
  fetch="select",另外发送一条select语句抓取当前对象关联实体或集合
2 设置fetch="jion"
  <many-to-one name="classes" column="classesid" cascade="save-update" fetch="join"/>
  fetch="jion",hibernate会通过select语句会使用外联接来加载其关联实体或集合,此时lazy会失效
------------------------------------------------------------
抓取策略(集合代理的批量抓取)
1 保持默认,也就是fetch="select"
  <set name="students" fetch="select">
  fetch="select",另外发送一条select语句抓取当前对象关联实体或集合
2 设置fetch="jion"
  <set name="students" fetch="jion">
  fetch="jion",hibernate会通过select语句会使用外联接来加载其关联实体或集合,此时lazy会失效
3 设置fetch="subselect"
  <set name="students" fetch="subselect">
  fetch="subselect",用于createQuery()查询,另外发送一条select语句抓取在前面查询到的所有实体对象的关联集合
----------------------------------------------------------------
抓取策略,batch-size在<class>上的应用

batch-size属性,可能批量加载体类,参见:Classes.hbm.xml
<class name="Classes" table="t_classes" batch-size="3">

在hibernate.cfg.xml中设置
<property name="hibernate.jdbc.fetch_size">50</property>

posted @ 2009-11-03 16:34 junly 阅读(765) | 评论 (0)编辑 收藏
     摘要: lazy策略可以使用在:    * <class>标签上,可以取值:true/false    * <property>标签上,可以取值:true/false需要类增强工具    * <set><list>标签上,可以取值:true/false/extra    ...  阅读全文
posted @ 2009-11-03 16:33 junly 阅读(524) | 评论 (0)编辑 收藏

Component映射(值对象映射)

在hibernate中,component是某个实体的逻辑组成部分,它与实体的根本区别是没有oid,
component可以称为是值对象(DDD)

采用component映射的好处:它实现了对象模型的细粒度划分,层次会更分明,复用率会更高

<!--
  User:                             Comtact:
  private int id;                    private String email;
    private String name;         private String address;
    private Comtact comtact;     private String phone;
-->
<class name="User" table="t_user">
        
<id name="id">
            
<generator class="native"/>
        
</id>
        
<property name="name"/>
        
<component name="comtact">
            
<property name="email"/>
            
<property name="address"/>
            
<property name="phone"/>
        
</component>
</class>

复合(联合)主键映射

通常将复合主键相关的属性,单独放到一个类中
 * 此类必须实现序列化接口
 * 覆写hashcode和equals方法

<class name="com.bjsxt.hibernate.FiscalYearPeriod" table="t_fiscal_year_period">
        
<composite-id name="fiscalYearPeriodPK">
            
<key-property name="fiscalYear"/>
            
<key-property name="fiscalPeriod"/>    
        
</composite-id>
        
<property name="beginDate"/>
        
<property name="endDate"/>
        
<property name="periodSts"/>
    
</class>

public class FiscalYearPeriodPK implements Serializable {
    
//核算年
    private int fiscalYear;
    
//核算月
    private int fiscalPeriod;
    
public int getFiscalYear() {
        
return fiscalYear;
    }

    
public void setFiscalYear(int fiscalYear) {
        
this.fiscalYear = fiscalYear;
    }

    
public int getFiscalPeriod() {
        
return fiscalPeriod;
    }

    
public void setFiscalPeriod(int fiscalPeriod) {
        
this.fiscalPeriod = fiscalPeriod;
    }


    @Override
    
public int hashCode() {
        
final int prime = 31;
        
int result = 1;
        result 
= prime * result + fiscalPeriod;
        result 
= prime * result + fiscalYear;
        
return result;
    }


    @Override
    
public boolean equals(Object obj) {
        
if (this == obj)
            
return true;
        
if (obj == null)
            
return false;
        
if (getClass() != obj.getClass())
            
return false;
        
final FiscalYearPeriodPK other = (FiscalYearPeriodPK) obj;
        
if (fiscalPeriod != other.fiscalPeriod)
            
return false;
        
if (fiscalYear != other.fiscalYear)
            
return false;
        
return true;
    }

}


public class FiscalYearPeriod {
    
private FiscalYearPeriodPK fiscalYearPeriodPK;
    
//开始日期
    private Date beginDate;
    
//结束日期
    private Date endDate;
    
//状态
    private String periodSts;
}
posted @ 2009-11-03 16:22 junly 阅读(187) | 评论 (0)编辑 收藏
1 class Node:
private int id;
private String name;
private Node parent;//交节点
private Set children;//子节点
2 Node.hbm.xml:
<class name="node" class="com.my.hibernate.Node">
        
<id name="id">
                
<generator class="native"/>
        
</id>
        
<property name="name"/>
        
<many-to-one name="parent" column="pid"/>
        
<set name="children" lazy="false" inverse="true"  cascade="all">
                
<key column="pid"/>
                
<one-to-many class="com.my.hibernate.Node"/>
        
</set>
</class>
posted @ 2009-11-03 16:19 junly 阅读(586) | 评论 (1)编辑 收藏
set、list、array、map

Collection:
 private int id;
 private String name;
 private Set setValue;
 private List listValue;
 private String[] arrayValue;
 private Map mapValue;
<class name="Collection">
        
<id name="id">
            
<generator class="native"/>
        
</id>
        
<property name="name"/>
        
<set name="setValue" table="t_setvalue">
            
<key column="setid"/>
            
<element type="string" column="setvalue"/>
        
</set>
        
<list name="listValue" table="t_listvalue">
            
<key column="listid"/>
            
<list-index column="listindex"/>
            
<element type="string" column="listvalue"/>
        
</list>
        
<array name="arrayValue" table="t_arrayvalue">
            
<key column="arrayid"/>
            
<list-index column="arrayindex"/>
            
<element type="string" column="arrayvalue"/>
        
</array>
        
<map name="mapValue" table="t_mapvalue">
            
<key column="mapid"/>
            
<map-key type="string" column="mapkey"/>
            
<element type="string" column="mapvalue"/>
        
</map>
    
</class>
posted @ 2009-11-03 16:17 junly 阅读(168) | 评论 (0)编辑 收藏
     摘要: 继承映射的三种策略:   * 单表继承,每棵类继承树使用一个表   * 具体表继承,每个子类一个表   * 类表继承,每个具体类一个表 ----------------------------------------------------------------- 每棵类继承树映射成一张表 1、理解如何映射   因为类继承树肯定是对...  阅读全文
posted @ 2009-11-03 16:15 junly 阅读(180) | 评论 (0)编辑 收藏
仅列出标题
共18页: First 上一页 10 11 12 13 14 15 16 17 18 下一页