详解Hibernate 照射关系
详解Hibernate 映射关系
Hibernate的映射关联关系和我们现实世界里事物的关联关系一样.比如在UML语言中,以客户Customer和订单Order的关系为例.一个客 户可以发送多个订单,而一个订单只能属于一个客户,这是一对多的关联,因此可以成为单向关联.如果同时包含了两两种关联关系,就成为双向关联.在关系数据 库中只有外键参照主键的关系.所以关系数据库实际上至支持一对一,或一对多的单向关系.在类于类之间的关系中.要算多对一关系和数据库中的外键参照主键关 系最匹配了.因此如果使用单向关联从订单到客户的多对一单向关联,在订单类中就要定义一个客户的属性.表示这个订单属于哪个客户,而客户类就无需定义存放 订单的集合属性了.下面写一个简单的例子.
//首先定义客户类
public class Customer implements Sreializable {
private Long id;
private String name;
//省略属性的访问方法
}
//然后定义订单类
public class Order implements Sreializable {
private Long id;
private String orderName;
private Customer customer;
//省略属性的访问方法,要注意的是Customer的访问方法.
}
Customer类的所有属性和CUSTOMERS表的所有属性一一对应,创建起来就比较简单了.下面主要看一下Order类的映射文件.
< property name=\"orderName\" type=\"string\" >
< column name=\"ORDER_NAME\" length=\"15\"/ >
< /property >
因为customer属性是是Customer类型,而ORDERS表的CUSTOMER_ID是整数类型,是不匹配的.所以我们不能用普通的元素来定义,而我们需要使用元素来配置了.
< many-to-one name=\"customer\" column=\"CUSTOMER_ID\" class=\"包名.Customer\" not-null=\"true\"/ >
< many-to-one >元素负责建立Order订单类的customer属性和数据库中的CUSTOMER_ID外键字段之间的映射.
name:设定映射文件的属性名
column:设定和持久化类对应的表的外键名
class:设定持久化类的属性的类型,这里指定具体的类,也就是主键存在的类
not-null:设定为true表示customer属性不允许为null,默认是false,这个属性会影响到bhm2ddl工具,会为 ORDERS表的CUSTOMER_ID外键设置为不允许空的约束,但是不会影响到hbm2java工具生长java源代码.此外还会影响到 Hibernate运行时的行为,在保存Order对象的时候会检查customer属性是否为null.用hbm2ddl编译之后得到的数据库文件如 下:
create table CUSTOMERS (
ID bigint not null,
NAME varchar(15),
primary key (ID)
);
create table ORDERS (
ID bigint not null,
ORDER_NUMBER varchar(15),
CUSTOMER_ID bigint not null,
primary key (ID)
);
alter table ORDERS add index FK8B7256E516B4891C (CUSTOMER_ID),
add constraint FK8B7256E516B4891C foreign key (CUSTOMER_ID) references CUSTOMERS (ID);
看到结果我们可以简单的把理解为在数据库中,创建外键的作用.上边这个例子就简单的演示了Hibernate映射 的一对一关联关系,至于一对多的关联关系比这个稍微复杂一点.而且可以看出,当Hibernate持久化一个临时对象的时候,在默认的情况下它不会自动持 久化关联其他临时对象,而是会抛出TransientObjectException异常.如果希望Hibernate持久化对象的时候也自动持久化说关 联的对象,就要把元素的cascade属性设置为save-update,表示级联操作的意思,cascade属性的默认值为none.当这个属性设置OK了.数据库就实现了级联保存更新的操作.
在类和类之间建好了关联关系之后,就可以方便的从一个对象得到它关联的对象.例如Customer customer=order.getCustomer();这样获得的了Customer对象了.但是如果想获得所有属于Customer客户的 Order订单对象,就涉及到了一对多双向关联了.在内存中,从一个对象导航都另一个对象要比从数据库中通过一个字段查询另一个字段快的多的多,但是也给 编程的时候带来了麻烦,随意修改一个对象就可能牵一发而动全身,所以说双向的关联比较复杂,但是类和类之间到底建立单向还是双向关联,这个要根据业务需求 来决定.比如说业务需求根据指定客户查询客户所有订单,根据指定的订单,查询出发这个订单的客户.这个时候我们不妨用多对一双向关联处理.其实上边的例子 的映射文件已经简历了客户和订单之间的一对多双向关联关系,只不过要在客户类中加一个集合的属性: [Page]
private set orders = new HashSet();
public set getOrders() {
return orders;
}
public void setOrders(Set orders) {
this.orders = orders;
}
有了orders属性,客户就可以通过getOrders()方法或者客户的全部订单了,Hibernate在定义这个集合属性的时候必须声 明为接口类型,但是不光光是Set还有Map和List,这样可以提高程序的强壮性,就是说set方法接受的对象只要是实现了Set接口就OK.避免出现 null值的现象.这里要注意的是hbm2java工具生成类的集合属性的代码时,不会给它初始化一个集合对象的实例,这里我们需要自己手动修改,当然不 修改也是可以的.接下来还要在customer.hbm.xml映射文件里映射集合类型的orders属性,当然这个和order表的的同理,所以不能通过普通的元素来设置属性和字段的映射关系.要使用元素来设置:
< set name=\"orders\" cascade=\"save-update\" >
< key column=\"CUSTOMER_ID\">
< one-to-many class=\"包名.Order\" >
< /set >
name:设定类的属性名
cascade:设置为save-update表示级联保存更新,当保存或更新Customer类的时候会级联保存更新跟它关联的Order类.
< key >元素是用来设定跟持久化类关联的类的外键
< one-to-many >元素看起来很熟悉,哦是设置外键的元素反过来了.这里它是用来设置所关联的持久化类的.这里设置为和客户关联的订单Order类,这里表明这个属性里要存放一组Order类型的对象.
这个< set >元素是表示orders属性声明为set类型.
< set >元素还有一个inverse属性,这个方法主要是在给已存在数据库中的字段建立关联的时候很有用.就是说当我们获得数据库中的两个表的两条记录的 对象customer客户对象和order订单对象(映射文件已经建立了他们类和类之间的关联,但外键的值为null的情况下)然后我们想建立这个客户对 象和订单对象之间的关联,我们要先调用order.setCustomer(customer);然后在调用 customer.getOrder().add(order);在Hibernate自动清理缓存的持久化对象的时候会提交两条SQL语句.进行了两个 update操作.但是实际上只修改了一条记录.重复的执行SQL语句是会降低系统的运行效率的,当把inverse属性设置为true的时候,同样的操 作就会合并到一条SQL语句执行了,inverse默认为false;
级联删除就很简单了,把cascade属性设置为delete,如果你删除了一个客户,程序就会先执行删除这个客户全部的订单的SQL语句,然后在删除这 个客户,所谓删除一个持久化对象不是在内存中删除这个对象,而是删除数据库中相关的记录,这个对象依然在内存中,只不过由持久化状态转为临时状态,当这个 对象的引用消失后,这个对象会被垃圾回收.但是如果我又想级联删除,还想级联保存,更新的时候应该怎么办呢?这个时候我们将cascade属性设置为 all-delete-orphan就OK了.非常简单明了.我们还可以通过持久化类的 customer.getOrder().rumove(order);解除关联.这里的操作表示获得客户订单的集合对象,然后从集合对象中删除 order的订单,其实这种操作的意义不大,当我们不需要的这个订单的时候完全可以删除它,解除关联之后如果设置了级联删除属性,这个无用的记录也是要被 删除的.其实解除关联就是把外键设为null.通常我们的外键都要约束不可以为空.
映射关联还有一种多对多的关联,是一种自身关联关系.就是同一张表.自己和自己的关联.比如说一张人表,地球人是人,美国人,中国人,日本人 都属于地球人,中国人有分北京人,山东人.日本人也有下一级的比如东京人.下面设想如果日本人被消灭掉了,那么东京人也应该都被没有了吧,这就是一种关 系,自身对自身的关联关系.这就有点类似树的结构了.下面用一个例子演示这种关系,代码来源于孙MM的<<精通 Hibernate>>一书.
public class Category implements Serializable {
private Long id;
private String name;
private Category parentCategory;
private Set childCategories; [Page]
public Category(String name, mypack.Category parentCategory, Set childCategories) {
this.name = name;
this.parentCategory = parentCategory;
this.childCategories = childCategories;
}
public Category() {
}
public Category(Set childCategories) {
this.childCategories = childCategories;
}
public Category getParentCategory() {
return this.parentCategory;
}
public void setParentCategory(Category parentCategory) {
this.parentCategory = parentCategory;
}
public Set getChildCategories() {
return this.childCategories;
}
public void setChildCategories(Set childCategories) {
this.childCategories = childCategories;
}
//为了节省空间省略了id,name属性的访问方法
}
< hibernate-mapping >
< class name=\"mypack.Category\" table=\"CATEGORIES\" >
< id name=\"id\" type=\"long\" column=\"ID\" >
< generator class=\"increment\"/ >
< /id >
< property name=\"name\" type=\"string\" >
< column name=\"NAME\" length=\"15\" / >
< /property>
< set
name=\"childCategories\"
cascade=\"save-update\"
inverse=\"true\"
>
< key column=\"CATEGORY_ID\" / >
< one-to-many class=\"mypack.Category\" / >
< /set >
< many-to-one
name=\"parentCategory\"
column=\"CATEGORY_ID\"
class=\"mypack.Category\"
/ >
< /class >
< /hibernate-mapping >
我觉得这种方式其实和上边的一对多,一对一关系一样,只不过两个用的都是同一个类罢了.看一下例子理解上应该很简单。
Hibernate的映射关联关系和我们现实世界里事物的关联关系一样.比如在UML语言中,以客户Customer和订单Order的关系为例.一个客 户可以发送多个订单,而一个订单只能属于一个客户,这是一对多的关联,因此可以成为单向关联.如果同时包含了两两种关联关系,就成为双向关联.在关系数据 库中只有外键参照主键的关系.所以关系数据库实际上至支持一对一,或一对多的单向关系.在类于类之间的关系中.要算多对一关系和数据库中的外键参照主键关 系最匹配了.因此如果使用单向关联从订单到客户的多对一单向关联,在订单类中就要定义一个客户的属性.表示这个订单属于哪个客户,而客户类就无需定义存放 订单的集合属性了.下面写一个简单的例子.
//首先定义客户类
public class Customer implements Sreializable {
private Long id;
private String name;
//省略属性的访问方法
}
//然后定义订单类
public class Order implements Sreializable {
private Long id;
private String orderName;
private Customer customer;
//省略属性的访问方法,要注意的是Customer的访问方法.
}
Customer类的所有属性和CUSTOMERS表的所有属性一一对应,创建起来就比较简单了.下面主要看一下Order类的映射文件.
< property name=\"orderName\" type=\"string\" >
< column name=\"ORDER_NAME\" length=\"15\"/ >
< /property >
因为customer属性是是Customer类型,而ORDERS表的CUSTOMER_ID是整数类型,是不匹配的.所以我们不能用普通的元素来定义,而我们需要使用元素来配置了.
< many-to-one name=\"customer\" column=\"CUSTOMER_ID\" class=\"包名.Customer\" not-null=\"true\"/ >
< many-to-one >元素负责建立Order订单类的customer属性和数据库中的CUSTOMER_ID外键字段之间的映射.
name:设定映射文件的属性名
column:设定和持久化类对应的表的外键名
class:设定持久化类的属性的类型,这里指定具体的类,也就是主键存在的类
not-null:设定为true表示customer属性不允许为null,默认是false,这个属性会影响到bhm2ddl工具,会为 ORDERS表的CUSTOMER_ID外键设置为不允许空的约束,但是不会影响到hbm2java工具生长java源代码.此外还会影响到 Hibernate运行时的行为,在保存Order对象的时候会检查customer属性是否为null.用hbm2ddl编译之后得到的数据库文件如 下:
create table CUSTOMERS (
ID bigint not null,
NAME varchar(15),
primary key (ID)
);
create table ORDERS (
ID bigint not null,
ORDER_NUMBER varchar(15),
CUSTOMER_ID bigint not null,
primary key (ID)
);
alter table ORDERS add index FK8B7256E516B4891C (CUSTOMER_ID),
add constraint FK8B7256E516B4891C foreign key (CUSTOMER_ID) references CUSTOMERS (ID);
看到结果我们可以简单的把理解为在数据库中,创建外键的作用.上边这个例子就简单的演示了Hibernate映射 的一对一关联关系,至于一对多的关联关系比这个稍微复杂一点.而且可以看出,当Hibernate持久化一个临时对象的时候,在默认的情况下它不会自动持 久化关联其他临时对象,而是会抛出TransientObjectException异常.如果希望Hibernate持久化对象的时候也自动持久化说关 联的对象,就要把元素的cascade属性设置为save-update,表示级联操作的意思,cascade属性的默认值为none.当这个属性设置OK了.数据库就实现了级联保存更新的操作.
在类和类之间建好了关联关系之后,就可以方便的从一个对象得到它关联的对象.例如Customer customer=order.getCustomer();这样获得的了Customer对象了.但是如果想获得所有属于Customer客户的 Order订单对象,就涉及到了一对多双向关联了.在内存中,从一个对象导航都另一个对象要比从数据库中通过一个字段查询另一个字段快的多的多,但是也给 编程的时候带来了麻烦,随意修改一个对象就可能牵一发而动全身,所以说双向的关联比较复杂,但是类和类之间到底建立单向还是双向关联,这个要根据业务需求 来决定.比如说业务需求根据指定客户查询客户所有订单,根据指定的订单,查询出发这个订单的客户.这个时候我们不妨用多对一双向关联处理.其实上边的例子 的映射文件已经简历了客户和订单之间的一对多双向关联关系,只不过要在客户类中加一个集合的属性: [Page]
private set orders = new HashSet();
public set getOrders() {
return orders;
}
public void setOrders(Set orders) {
this.orders = orders;
}
有了orders属性,客户就可以通过getOrders()方法或者客户的全部订单了,Hibernate在定义这个集合属性的时候必须声 明为接口类型,但是不光光是Set还有Map和List,这样可以提高程序的强壮性,就是说set方法接受的对象只要是实现了Set接口就OK.避免出现 null值的现象.这里要注意的是hbm2java工具生成类的集合属性的代码时,不会给它初始化一个集合对象的实例,这里我们需要自己手动修改,当然不 修改也是可以的.接下来还要在customer.hbm.xml映射文件里映射集合类型的orders属性,当然这个和order表的的同理,所以不能通过普通的元素来设置属性和字段的映射关系.要使用元素来设置:
< set name=\"orders\" cascade=\"save-update\" >
< key column=\"CUSTOMER_ID\">
< one-to-many class=\"包名.Order\" >
< /set >
name:设定类的属性名
cascade:设置为save-update表示级联保存更新,当保存或更新Customer类的时候会级联保存更新跟它关联的Order类.
< key >元素是用来设定跟持久化类关联的类的外键
< one-to-many >元素看起来很熟悉,哦是设置外键的元素反过来了.这里它是用来设置所关联的持久化类的.这里设置为和客户关联的订单Order类,这里表明这个属性里要存放一组Order类型的对象.
这个< set >元素是表示orders属性声明为set类型.
< set >元素还有一个inverse属性,这个方法主要是在给已存在数据库中的字段建立关联的时候很有用.就是说当我们获得数据库中的两个表的两条记录的 对象customer客户对象和order订单对象(映射文件已经建立了他们类和类之间的关联,但外键的值为null的情况下)然后我们想建立这个客户对 象和订单对象之间的关联,我们要先调用order.setCustomer(customer);然后在调用 customer.getOrder().add(order);在Hibernate自动清理缓存的持久化对象的时候会提交两条SQL语句.进行了两个 update操作.但是实际上只修改了一条记录.重复的执行SQL语句是会降低系统的运行效率的,当把inverse属性设置为true的时候,同样的操 作就会合并到一条SQL语句执行了,inverse默认为false;
级联删除就很简单了,把cascade属性设置为delete,如果你删除了一个客户,程序就会先执行删除这个客户全部的订单的SQL语句,然后在删除这 个客户,所谓删除一个持久化对象不是在内存中删除这个对象,而是删除数据库中相关的记录,这个对象依然在内存中,只不过由持久化状态转为临时状态,当这个 对象的引用消失后,这个对象会被垃圾回收.但是如果我又想级联删除,还想级联保存,更新的时候应该怎么办呢?这个时候我们将cascade属性设置为 all-delete-orphan就OK了.非常简单明了.我们还可以通过持久化类的 customer.getOrder().rumove(order);解除关联.这里的操作表示获得客户订单的集合对象,然后从集合对象中删除 order的订单,其实这种操作的意义不大,当我们不需要的这个订单的时候完全可以删除它,解除关联之后如果设置了级联删除属性,这个无用的记录也是要被 删除的.其实解除关联就是把外键设为null.通常我们的外键都要约束不可以为空.
映射关联还有一种多对多的关联,是一种自身关联关系.就是同一张表.自己和自己的关联.比如说一张人表,地球人是人,美国人,中国人,日本人 都属于地球人,中国人有分北京人,山东人.日本人也有下一级的比如东京人.下面设想如果日本人被消灭掉了,那么东京人也应该都被没有了吧,这就是一种关 系,自身对自身的关联关系.这就有点类似树的结构了.下面用一个例子演示这种关系,代码来源于孙MM的<<精通 Hibernate>>一书.
public class Category implements Serializable {
private Long id;
private String name;
private Category parentCategory;
private Set childCategories; [Page]
public Category(String name, mypack.Category parentCategory, Set childCategories) {
this.name = name;
this.parentCategory = parentCategory;
this.childCategories = childCategories;
}
public Category() {
}
public Category(Set childCategories) {
this.childCategories = childCategories;
}
public Category getParentCategory() {
return this.parentCategory;
}
public void setParentCategory(Category parentCategory) {
this.parentCategory = parentCategory;
}
public Set getChildCategories() {
return this.childCategories;
}
public void setChildCategories(Set childCategories) {
this.childCategories = childCategories;
}
//为了节省空间省略了id,name属性的访问方法
}
< hibernate-mapping >
< class name=\"mypack.Category\" table=\"CATEGORIES\" >
< id name=\"id\" type=\"long\" column=\"ID\" >
< generator class=\"increment\"/ >
< /id >
< property name=\"name\" type=\"string\" >
< column name=\"NAME\" length=\"15\" / >
< /property>
< set
name=\"childCategories\"
cascade=\"save-update\"
inverse=\"true\"
>
< key column=\"CATEGORY_ID\" / >
< one-to-many class=\"mypack.Category\" / >
< /set >
< many-to-one
name=\"parentCategory\"
column=\"CATEGORY_ID\"
class=\"mypack.Category\"
/ >
< /class >
< /hibernate-mapping >
我觉得这种方式其实和上边的一对多,一对一关系一样,只不过两个用的都是同一个类罢了.看一下例子理解上应该很简单。