日期:2014-05-16  浏览次数:20556 次

详解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类型.

  &l