hibernate映射---多对一(list)

在学校宿舍里面,床和宿舍(tb_bed\tb_room)就是多对一的关系。

cascade的属性:

级联指的是当主控方执行操作时,关联对象(被动方)是否同步执行同一操作。pojo和它的关系属性的关系就是“主控方 -- 被动方”的关系,如果关系属性是一个list,那么被动方就是list中的一个一个元素。

比如删除tb_room----session.delete(room);级联决定了是否执行关联到tb_room的tb_bed---session.delete(bed);

cascade="save-update": 级联保存(load以后如果子对象发生了更新,也会级联更新), 但它不会级联删除。一对一、一对多关系(首端为一)一般设置为save-update;

cascade="delete": 级联删除, 但不具备级联保存和更新(例如删除班级,将级联删除下属的学生);

all-delete-orphan: 在解除父子关系时,自动删除不属于父对象的子对象, 也支持级联删除和级联保存更新。主要用于从集合中删除对象的情况,此时被删除的对象将会被删除成为脱管对象。

all: 级联删除, 级联更新,但解除父子关系时不会自动删除子对象.;

none:多对一、多对多关系(首端为多)一般设置为none。

(1)表结构:如图片所示,tb_bed通过room_sn关联到tb_room中。

create table TB_BED
(
SN NUMBER not null,
ROOM_SN NUMBER,
NAME VARCHAR2(20),
USERNAME VARCHAR2(20),
BEDINDEX NUMBER
)

create table TB_ROOM
(
SN NUMBER,
NAME VARCHAR2(20),
ADDRESS VARCHAR2(50),
FLOOR VARCHAR2(2)
)

(2)映射文件

tb_room的映射文件如下:

<hibernate-mapping>
<class name="com.longtop.entity.TbRoom" table="TB_ROOM" schema="TEST">
<id name="sn" type="java.lang.Long" column="SN">
<generator class="sequence" >
<param name="sequence">seq_tb_room_sn</param>
</generator>
</id>
<property name="floor" type="string">
<column name="FLOOR" length="2"/>
</property>
<property name="name" type="string">
<column name="NAME" length="20" />
</property>
<property name="address" type="string">
<column name="ADDRESS" length="50" />
</property>
<list name="beds" table="TB_ROOM" lazy="false" inverse="true" cascade="all-delete-orphan" >
<key column="ROOM_SN" />
<index column="BEDINDEX"/>
<one-to-many class="com.longtop.entity.TbBed"/>
</list>
</class>
</hibernate-mapping>



注意list的子元素必须包含key,index,one-to-many等子元素,其中column都是tb_bed的列名。因为集合采用list对象,所以tb_bed必须增加一个字段index用于存放属于同一个tb_room的tb_bed集合的序号(与list的index相对应)。

tb_bed的映射文件如下:

<hibernate-mapping>
<class name="com.longtop.entity.TbBed" table="TB_BED" schema="TEST">
<id name="sn" type="java.lang.Long" column="SN">
<generator class="sequence" >
<param name="sequence">seq_tb_bed_sn</param>
</generator>
</id>
<property name="roomSn" type="long">
<column name="ROOM_SN" length="22"/>
</property>
<property name="name" type="string">
<column name="NAME" length="20" />
</property>
<property name="username" type="string">
<column name="USERNAME" length="20" />
</property>
<property name="bedindex" type="long">
<column name="BEDINDEX" length="20" />
</property>
</class>
</hibernate-mapping>



(3)tb_room、tb_bed的映射类如下:

public class TbRoom {
private Long sn;
private String name;
private String address;
private String floor;
private List beds = new ArrayList();
public List getBeds() {
return beds;
}
public void setBeds(List beds) {
this.beds = beds;
}
public Long getSn() {
return sn;
}
public void setSn(Long sn) {
this.sn = sn;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getAddress() {
return address;
}
public void setAddress(String address) {
this.address = address;
}
public String getFloor() {
return floor;
}
public void setFloor(String floor) {
this.floor = floor;
}
}



tb_bed的映射类代码如下:

public class TbBed {
private Long sn;
private Long roomSn;
private String username;
private String name;
private Long bedindex;
public Long getBedindex() {
return bedindex;
}
public void setBedindex(Long bedindex) {
this.bedindex = bedindex;
}
public Long getSn() {
return sn;
}
public void setSn(Long sn) {
this.sn = sn;
}

public Long getRoomSn() {
return roomSn;
}
public void setRoomSn(Long roomSn) {
this.roomSn = roomSn;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}



(3)测试类

(一)测试类的测试代码

public class testMain {

public static void main(String[] args) throws HibernateException {

SessionFactory sessions =
new Configuration().configure().buildSessionFactory();
Session session = sessions.openSession();
Transaction tx = session.beginTransaction();
Long start = new Date().getTime();
System.out.println("starttime:"+start);

// Long sn = testMain.createRoom(session);
TbRoom room = testMain.loadRoom(session, new Long(23));
testMain.deleteBed(session, null, room);

tx.commit();
session.close();

Long end = new Date().getTime();
System.out.println("endtime:"+end);
System.out.println("总共用了"+(end-start)+"的时间");
}
}

(二)添加记录的方法(添加tb_room记录并添加对应到该tb_room的tb_bed对象)。

private static Long createRoom(Session session){
TbRoom room = new TbRoom();
room.setName("102");
room.setFloor("1");
room.setAddress("公寓1号楼");

TbBed bed1 = new TbBed();
bed1.setName("1床");
bed1.setUsername("aimy");

TbBed bed2 = new TbBed();
bed2.setName("2床");
bed2.setUsername("Lucy");

TbBed bed3 = new TbBed();
bed3.setName("3床");
bed3.setUsername("judy");

TbBed bed4 = new TbBed();
bed4.setName("4床");
bed4.setUsername("rose");

try {
room.getBeds().add(bed1);
room.getBeds().add(bed2);
room.getBeds().add(bed3);
room.getBeds().add(bed4);
session.save(room);//只要保存一次tb_room,则自动保存对应的tb_bed
// session.save(bed1);
// session.save(bed2);
// session.save(bed3);
// session.save(bed4);
} catch (HibernateException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
return room.getSn();
}
}

在添加对象时,出现了以下的错误及情况:

(1)Exception in thread "main" java.lang.NullPointerException
at testMain.createRoom(testMain.java:102)
at testMain.main(testMain.java:30)

原因:testMain.java:102就是 room.getBeds().add(bed1); 行。room.getBeds()为null,所以没办法调用add方法,
解决的方法:必须在com.longtop.entity.TbRoom映射类中,声明tb_bed集合List beds时初始化。 private List beds = new ArrayList();

(2)如果tb_room的映射文件中list子元素如下设置:

<list name="beds" table="TB_ROOM" lazy="false" inverse="true" cascade="all-delete-orphan" >
<key column="ROOM_SN" />
<index column="BEDINDEX"/>
<one-to-many class="com.longtop.entity.TbBed"/>
</list>

运行添加的方法,则数据库增加了一条tb_room的记录和四条tb_bed记录,如下:

tb_room:

tb_bed:
注意:因为inverse="true",所以添加的tb_bed记录的room_sn没有关联到tb_room记录。如果inverse="false",则tb_bed记录自动通过room_sn关联到tb_room记录。

<list name="beds" table="TB_ROOM" lazy="false" inverse="false" cascade="all-delete-orphan" >
<key column="ROOM_SN" />
<index column="BEDINDEX"/>
<one-to-many class="com.longtop.entity.TbBed"/>
</list>

运行添加的方法,则数据库增加了一条tb_room的记录和四条tb_bed记录,如下:

tb_room:

tb_bed:

(三)加载tb_room记录以及相关的tb_bed记录

private static TbRoom loadRoom(Session session,Long sn)
throws HibernateException{
TbRoom room = (TbRoom) session.load(TbRoom.class,sn);
List beds = room.getBeds();
TbBed bed;
for(int i=0;i<beds.size();i++){
bed = (TbBed) beds.get(i);
if(bed!=null)
System.out.println("bed name: "+bed.getName());
}
return room;
}

数据库的数据如下:

tb_room:

tb_bed:

因为room_sn=4的tb_bed记录的bedindex=0,1,3,5,所有tb_room记录的beds集合就是按照这个顺序来排序的,因为没有bedindex没有bedindex=2,所以list.get(2)是空的。list对象如下图所示:

TbRoom的list beds对象: 所以取beds集合的TbBed对象时必须判断是否非空。

(四)删除tb_room记录以及相关的tb_bed记录

private static void deleteRoom(Session session,TbRoom room)
throws HibernateException{
System.out.println("删除room"+room.getName());
session.delete(room);
}

①删除tb_room记录但不删除相关的tb_bed记录,只是把关联性去掉。是否删除room对应的tb_bed记录的关键在于配置文件。

<list name="beds" table="TB_ROOM" lazy="false"
inverse="true" cascade="all-delete-orphan" >
<key column="ROOM_SN" />
<index column="BEDINDEX"/>
<one-to-many class="com.longtop.entity.TbBed"/>
</list>

删除记录之前的tb_bed表如图所示



如果inverse="true",运行session.delete(room); 则sn=7的tb_room记录删除, 而room_sn=21的tb_bed记录没有删除,只是将room_sn,bedindex清空。



②删除tb_room记录及删除关联到他的tb_bed集合。inverse必须设为false.删除了某一条记录后,属于同一个room内的tb_bed序号>这条记录的序号(bedindex)都减1.

<list name="beds" table="TB_ROOM" lazy="false"
inverse="false" cascade="all-delete-orphan" >
<key column="ROOM_SN" />
<index column="BEDINDEX"/>
<one-to-many class="com.longtop.entity.TbBed"/>
</list>



③删除tb_room的某一条tb_bed记录。

private static void deleteBed(Session session,TbBed bed,TbRoom room)
throws HibernateException{
System.out.println("删除room床号0床位!");
room.getBeds().remove(0);
session.save(room);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值