新手学习Hiberbate必读 ------笔记

第一个hibernate项目:

1、新建java项目

2、创建User Library,加入如下jar

* HIBERNATE_HOME/hibernate3.jar

* HIBERNATE_HOME/lib/*.jar

* MySql jdbc驱动

3、创建hibernate配置文件hibernate.cfg.xml,为了便于调试最好加入log4j配置文件

4、定义实体类

5、定义User类的映射文件User.hbm.xml

6、将User.hbml.xml文件加入到hibernate.cfg.xml文件中

7、编写hbm2ddl工具类,将实体类生成数据库表

8、开发客户端

为了方便跟踪sql执行,在hibernate.cfg.xml文件中加入<property name="hibernate.show_sql">true</property>

session flush测试:

session flush方法主要做了两件事:

* 清理缓存

* 执行sql

session在什么情况下执行flush

* 默认在事务提交时

* 显示的调用flush

* 在执行查询前,如:iterate

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

测试实体对象的生命周期:

junit简介:

* 编写测试类xxxTest,需要继承TestCase

* 编写单元测试方法,测试方法必须以test开头,测试方法不能含有参数和返回值,如:

public void testHello1() {}

* 最好单元测试的代码单独建立一个目录

了解Hibernate中CRUD操作

了解get和load的区别?

* get不支持lazy,load支持lazy

* 采用get加载数据,如果没有匹配的数据,返回null,而load则抛出异常

transient状态的特征?

* 在数据库中没有与之匹配的数据

* 没有纳入session的管理

persistent状态的特征?

* persistent状态的对象在数据库中有与之匹配的数据

* 纳入了session的管理

* 在清理缓存(脏数据检查)的时候,会和数据库同步

detached状态的特征?

* 在数据库中有与之匹配的数据

* 没有纳入session的管理

hibernate基本映射:

实体类---表

实体类中的普通属性---表字段

采用<class>标签映射成数据库表,通过<property>标签将普通属性映射成表字段

所谓普通属性指不包括自定义类、集合和数组等

注意:如果实体类和实体类中的属性和sql中的关键字重复,必须采用table或column重新命名

实体类的设计原则:

* 实现一个默认的(即无参数的)构造方法(constructor)

* 提供一个标识属性(identifier property)(可选)

* 使用非final的类 (可选)

* 为持久化字段声明访问器(accessors)

主键生成策略:

uuid、native和assigned

hibernate一对一主键关联映射(单向关联Person---->IdCard):

一对一主键关联映射:让两个实体对象的id保持相同,这样可以避免多余的字段被创建

具体映射:

<id name="id">

<!-- person的主键来源idCard,也就是共享idCard的主键 -->

<generator class="foreign">

<param name="property">idCard</param>

</generator>

</id>

<property name="name"/>

<!-- one-to-one标签的含义,指示hibernate怎么加载它的关联对象,默认根据主键加载,

constrained="true", 表明当前主键上存在一个约束,person的主键作为外键参照了idCard

-->

<one-to-one name="idCard" constrained="true"/>

hibernate一对一主键关联映射(双向关联Person<---->IdCard):

需要在idcard映射文件中加入<one-to-one>标签指向person,指示hibernate如何加载person

默认根据主键加载



hibernate一对一唯一外键关联映射(单向关联Person---->IdCard):



一对唯一外键关联映射是多对一关联映射的特例



可以采用<many-to-one>标签,指定多的一端的unique=true,这样就限制了多的一端的多重性为一

通过这种手段映射一对一唯一外键关联

hibernate一对一唯一外键关联映射(双向关联Person<---->IdCard):

一对一唯一外键关联双向,需要在另一端(idcard),添加<one-to-one>标签,指示hibernate如何加载

其关联对象,默认根据主键加载person,外键关联映射中,因为两个实体采用的是person的外键维护的关系,

所以不能指定主键加载person,而要根据person的外键加载,所以采用如下映射方式:

<one-to-one name="person" property-ref="idCard"/>

hihernate一对多关联映射(单向Classes----->Student):

一对多关联映射利用了多对一关联映射原理



多对一关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是多指向一

一对多关联映射:在多的一端加入一个外键指向一的一端,它维护的关系是一指向多



也就是说一对多和多对一的映射策略是一样的,只是站的角度不同



在一一端维护关系的缺点:

* 如果将t_student表里的classesid字段设置为非空,则无法保存

* 因为不是在student这一端维护关系,所以student不知道是哪个班的,

所以需要发出多余的update语句来更新关系



hihernate一对多关联映射(双向Classes<----->Student):

一对多双向关联映射:

* 在一一端的集合上使用<key>,在对方表中加入一个外键指向一一端

* 在多一端采用<many-to-one>



注意:<key>标签指定的外键字段必须和<many-to-one>指定的外键字段一致,否则引用字段的错误



如果在”一“一端维护一对多关联关系,hibernate会发出多余的udpate语句,所以我们一般在多

的一端来维护关联关系



关于inverse属性:

inverse主要用在一对多和多对多双向关联上,inverse可以被设置到集合标签<set>上,

默认inverse为false,所以我们可以从”一“一端和”多“一端维护关联关系,

如果设置成inverse为true,则我们只能从多一端来维护关联关系



注意:inverse属性,只影响数据的存储,也就是持久化



inverse和cascade

* inverse是关联关系的控制方向

* cascade操作上的连锁反应

hibernate多对一关联映射:

关联映射的本质:

* 将关联关系映射到数据库,所谓的关联关系是对象模型在内存中的一个或多个引用



<many-to-one>会在多的一端加入一个外键,指向一的一端,这个外键是由<many-to-one>

中的column属性定义的,如果忽略了这个属性那么默认的外键与实体的属性一致



<many-to-one>标签的定义示例:

* <many-to-one name="group" column="groupid"/>



理解级联的含义?

* 是对象的连锁操作





hibernate多对多关联映射(单向User---->Role):

具体映射方式:

<set name="roles" table="t_user_role">

<key column="userid"/>

<many-to-many class="com.bjsxt.hibernate.Role" column="roleid"/>

</set>

hibernate多对多关联映射(双向User<---->Role):

映射方法:

<set name="roles" table="t_user_role">

<key column="userid"/>

<many-to-many class="com.bjsxt.hibernate.Role" column="roleid"/>

</set>

table属性值必须和单向关联中的table属性值一致

<key>中column属性值要与单向关联中的<many-to-many>标签中的column属性值一致

在<many-to-many>中的column属性值要与单向关联中<key>标签的column属性值一致



普通集合映射:

set

list

array

map

component映射:



在hibernate中,component是某个实体的逻辑组成部分,它与实体的根本区别是没有oid,

component可以成为是值对象(DDD)



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

复合(联合)主键映射:

通常将复合主键相关的属性,单独放到一个类中

* 此类必须实现序列化接口

* 覆写hashcode和equals方法



每棵继承树映射成一张表:



1、理解如何映射

因为类继承树肯定是对应多个类,要把多个类的信息存放在一张表中,必须有某种机制来区分哪些记录是属于哪个类的。

这种机制就是,在表中添加一个字段,用这个字段的值来进行区分。用hibernate实现这种策略的时候,有如下步骤:

父类用普通的<class>标签定义

在父类中定义一个discriminator,即指定这个区分的字段的名称和类型

如:<discriminator column=”XXX” type=”string”/>

子类使用<subclass>标签定义,在定义subclass的时候,需要注意如下几点:

Subclass标签的name属性是子类的全路径名

在Subclass标签中,用discriminator-value属性来标明本子类的discriminator字段(用来区分不同类的字段)

的值Subclass标签,既可以被class标签所包含(这种包含关系正是表明了类之间的继承关系),也可以与class标

签平行。 当subclass标签的定义与class标签平行的时候,需要在subclass标签中,添加extends属性,里面的值

是父类的全路径名称。子类的其它属性,像普通类一样,定义在subclass标签的内部。



2、理解如何存储

存储的时候hibernate会自动将鉴别字段值插入到数据库中,在加载数据的时候,hibernate能根据这个鉴别值

正确的加载对象



多态查询:在hibernate加载数据的时候能鉴别出正真的类型(instanceOf)



get支持多态查询

load只有在lazy=false,才支持多态查询

hql支持多态查询





每棵继承树映射成一张表





每个子类映射成一张表:



1、理解如何映射

这种策略是使用joined-subclass标签来定义子类的。父类、子类,每个类都对应一张数据库表。

在父类对应的数据库表中,实际上会存储所有的记录,包括父类和子类的记录;在子类对应的数据库表中,

这个表只定义了子类中所特有的属性映射的字段。子类与父类,通过相同的主键值来关联。实现这种策略的时候,

有如下步骤:

父类用普通的<class>标签定义即可

父类不再需要定义discriminator字段

子类用<joined-subclass>标签定义,在定义joined-subclass的时候,需要注意如下几点:

Joined-subclass标签的name属性是子类的全路径名

Joined-subclass标签需要包含一个key标签,这个标签指定了子类和父类之间是通过哪个字段来关联的。

如:<key column=”PARENT_KEY_ID”/>,这里的column,实际上就是父类的主键对应的映射字段名称。

Joined-subclass标签,既可以被class标签所包含(这种包含关系正是表明了类之间的继承关系),

也可以与class标签平行。 当Joined-subclass标签的定义与class标签平行的时候,需要在Joined-subclass

标签中,添加extends属性,里面的值是父类的全路径名称。子类的其它属性,像普通类一样,定义在joined-subclass标签的内部。



每个子类映射成一张表

每个具体类映射成一张表:

1、理解如何映射

这种策略是使用union-subclass标签来定义子类的。每个子类对应一张表,而且这个表的信息是完备的,

即包含了所有从父类继承下来的属性映射的字段(这就是它跟joined-subclass的不同之处,

joined-subclass定义的子类的表,只包含子类特有属性映射的字段)。实现这种策略的时候,有如下步骤:

父类用普通<class>标签定义即可

子类用<union-subclass>标签定义,在定义union-subclass的时候,需要注意如下几点:

Union-subclass标签不再需要包含key标签(与joined-subclass不同)

Union-subclass标签,既可以被class标签所包含(这种包含关系正是表明了类之间的继承关系),

也可以与class标签平行。 当Union-subclass标签的定义与class标签平行的时候,需要在Union-subclass

标签中,添加extends属性,里面的值是父类的全路径名称。子类的其它属性,像普通类一样,

定义在Union-subclass标签的内部。这个时候,虽然在union-subclass里面定义的只有子类的属性,

但是因为它继承了父类,所以,不需要定义其它的属性,在映射到数据库表的时候,依然包含了父类的所

有属性的映射字段。



注意:在保存对象的时候id是不能重复的(不能使用自增生成主键)



每个具体类映射成一张表

Lazy:

hibernate lazy策略可以使用在:

* <class>标签上,可以取值:true/false

* <property>标签上,可以取值:true/false需要类增强工具

* <set><list>标签上,可以取值:true/false/extra

* <one-to-one><many-to-one>单端关联上,可以取值:false/proxy/noproxy



lazy概念:只有真正使用该对象时,才会创建,对于hibernate而言,正真使用的时候才会发出sql



hibernate支持lazy策略只有在session打开状态下有效



<class>标签上的lazy特性只对普通属性起作用



hibernate在集合上的lazy策略,可以取值:true/false/extra



<class>标签上的lazy不会影响到集合上的lazy特性



hibernate在单端关联上的lazy策略,可以取值:false/proxy/noproxy



<class>标签上的lazy不会影响到单端关联上的lazy特性



乐观锁:



大多数基于数据版本记录机制(version)实现,一般是在数据库表中加入一个version字段

读取数据时将版本号一同读出,之后更新数据时版本号加一,如果提交数据时版本号小于或等于数据表中

的版本号,则认为数据是过期的,否则给予更新

悲观锁:



悲观锁的实现,通常依赖于数据库机制,在整个过程中将数据锁定,其它任何用户都不能读取或修改



hibernate查询语言hql:



在hql中关键字不区分大小写,但是属性和类名区分大小写



1、简单属性查询【重要】

* 单一属性查询,返回结果集属性列表,元素类型和实体类中相应的属性类型一致

* 多个属性查询,返回的集合元素是对象数组,数组元素的类型和对应的属性在实体类中的类型一致

数组的长度取决与select中属性的个数

* 如果认为返回数组不够对象化,可以采用hql动态实例化Student对象

参见:SimplePropertyQueryTest.java



2、实体对象查询【重要】

* N + 1问题,在默认情况下,使用query.iterate查询,有可以能出现N+1问题

所谓的N+1是在查询的时候发出了N+1条sql语句

1: 首先发出一条查询对象id列表的sql

N: 根据id列表到缓存中查询,如果缓存中不存在与之匹配的数据,那么会根据id发出相应的sql语句

* list和iterate的区别?

* list每次都会发出sql语句,list会向缓存中放入数据,而不利用缓存中的数据

* iterate:在默认情况下iterate利用缓存数据,但如果缓存中不存在数据有可以能出现N+1问题

参见:SimpleObjectQueryTest1.java/SimpleObjectQueryTest2.java



3、条件查询【重要】

* 可以采用拼字符串的方式传递参数

* 可以采用 ?来传递参数(索引从0开始)

* 可以采用 :参数名 来传递参数

* 如果传递多个参数,可以采用setParamterList方法

* 在hql中可以使用数据库的函数,如:date_format

参见:SimpleConditionQueryTest.java



4、hibernate也支持直接使用sql进行查询

参见:SqlQueryTest.java



5、外置命名查询

* 在映射文件中采用<query>标签来定义hql

* 在程序中采用session.getNamedQuery()方法得到hql查询串

参见:Student.hbm.xml、NameQueryTest.java



6、查询过滤器

* 在映射文件中定义过滤器参数

* 在类的映射中使用这些参数

* 在程序中启用过滤器

参见:Student.hbm.xml、FilterQueryTest.java



7、分页查询【重要】

* setFirstResult(),从0开始

* setMaxResults,每页显示多少条数据

参见:PageQueryTest.java



8、对象导航查询,在hql中采用 . 进行导航【重要】

参见:ObjectNavQueryTest.java



9、连接查询【重要】

* 内连

* 外连接(左连接/右连接)

参见:JoinQueryTest.java



10、统计查询【重要】

参见:StatQueryTest.java



11、DML风格的操作(尽量少用,因为和缓存不同步)

参见:DMLQueryTest.java



hibernate一级缓存



一级缓存很短和session的生命周期一致,一级缓存也叫session级的缓存或事务级缓存



那些方法支持一级缓存:

* get()

* load()

* iterate(查询实体对象)



如何管理一级缓存:

* session.clear(),session.evict()



如何避免一次性大量的实体数据入库导致内存溢出

* 先flush,再clear



如果数据量特别大,考虑采用jdbc实现,如果jdbc也不能满足要求可以考虑采用数据本身的特定导入工具
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值