您的位置:首页 > 其它

Hibernate的双向1-N关联(五)

2016-03-08 23:48 399 查看
对于1-N关联,Hibernate推荐使用双向关联,而且不要让1的一端控制关联关系,而使用N的一端控制关联关系。双向的1-N关联与N-1关联是完全相同的两种情形,两端都需要增加对关联属性的访问,N的一端增加引用到关联实体的属性,1的一端增加集合属性,集合元素为关联关联实体。

1. 无连接表的双向1-N关联

无连接表的双向1-N关联,N的一端需要增加@ManyToOne注解来修饰代表关联实体的属性,而1的一端则需要使用@OneToMany注解来修饰代表关联实体的属性,且不能控制关联关系。而且使用@OneToMany注解时指定mappedBy属性——一旦为@OneToMany、@ManyToMany指定了该属性,则表明当前实体不能控制关联关系,这时Hibernate也不允许使用@JoinColumn或@JoinTable修饰代表关联实体的属性了。如下是Person类。

@Entity
@Table(name="person_inf")
public class Person
{
// 标识属性
@Id @Column(name="person_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private Integer id;
private String name;
private int age;
// 定义该Person实体所有关联的Address实体
// 指定mappedBy属性表明该Person实体不控制关联关系
@OneToMany(targetEntity=Address.class
, mappedBy="person")
private Set<Address> addresses
= new HashSet<>();

// id的setter和getter方法
public void setId(Integer id)
{
this.id = id;
}
public Integer getId()
{
return this.id;
}

// name的setter和getter方法
public void setName(String name)
{
this.name = name;
}
public String getName()
{
return this.name;
}

// age的setter和getter方法
public void setAge(int age)
{
this.age = age;
}
public int getAge()
{
return this.age;
}

// addresses的setter和getter方法
public void setAddresses(Set<Address> addresses)
{
this.addresses = addresses;
}
public Set<Address> getAddresses()
{
return this.addresses;
}
}


下面是Address的类。

@Entity
@Table(name="address_inf")
public class Address
{
// 标识属性
@Id @Column(name="address_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private int addressId;
// 定义地址详细信息的成员变量
private String addressDetail;
// 定义该Address实体关联的Person实体
@ManyToOne(targetEntity=Person.class)
// 定义名为person_id外键列,该外键列引用person_inf表的person_id列。
@JoinColumn(name="person_id" , referencedColumnName="person_id"
, nullable=false)
private Person person;

// 无参数的构造器
public Address()
{
}
// 初始化全部成员变量的构造器
public Address(String addressDetail)
{
this.addressDetail = addressDetail;
}

// addressId的setter和getter方法
public void setAddressId(int addressId)
{
this.addressId = addressId;
}
public int getAddressId()
{
return this.addressId;
}

// addressDetail的setter和getter方法
public void setAddressDetail(String addressDetail)
{
this.addressDetail = addressDetail;
}
public String getAddressDetail()
{
return this.addressDetail;
}

// person的setter和getter方法
public void setPerson(Person person)
{
this.person = person;
}
public Person getPerson()
{
return this.person;
}
}

我们用如下的类,来保存Person与Address的类。

public class PersonManager
{
public static void main(String[] args)
{
PersonManager mgr = new PersonManager();
mgr.testPerson();
HibernateUtil.sessionFactory.close();
}

private void testPerson()
{
Session session = HibernateUtil.currentSession();
Transaction tx = session.beginTransaction();
// 创建一个瞬态的Person对象
Person p = new Person();
// 设置Person的name为owen.org字符串
p.setName("owen.org");
p.setAge(29);
// 持久化Person对象(对应于插入主表记录)
session.save(p);
// 创建一个瞬态的Address对象
Address a = new Address("广州天河");
// 先设置Person和Address之间的关联关系
a.setPerson(p);
// 再持久化Address对象(对应于插入从表记录)
session.persist(a);
// 创建一个瞬态的Address对象
Address a2 = new Address("上海虹口");
// 先设置Person和Address之间的关联关系
a2.setPerson(p);
// 再持久化Address对象(对应于插入从表记录)
session.persist(a2);
tx.commit();
HibernateUtil.closeSession();
}

}

后台的SQL语句:

Hibernate:
insert
into
person_inf
(age, name)
values
(?, ?)
Hibernate:
insert
into
address_inf
(addressDetail, person_id)
values
(?, ?)
Hibernate:
insert
into
address_inf
(addressDetail, person_id)
values
(?, ?)

执行的结果如下:



2. 有连接表的双向1-N关联

Address类则需要使用@JoinTable映射连接表。

@Entity
@Table(name="address_inf")
public class Address
{
// 标识属性
@Id @Column(name="address_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private int addressId;
// 定义地址详细信息的成员变量
private String addressDetail;
// 定义该Address实体关联的Person实体
@ManyToOne(targetEntity=Person.class)
// 映射连接表,指定连接表为person_address
@JoinTable(name="person_address",
// 指定连接表中address_id列参照当前实体对应数据表的address_id主键列
joinColumns=@JoinColumn(name="address_id"
, referencedColumnName="address_id", unique=true),
// 指定连接表中person_id列参照当前实体的关联实体对应数据表的person_id主键列
inverseJoinColumns=@JoinColumn(name="person_id"
, referencedColumnName="person_id")
)
private Person person;

// 无参数的构造器
public Address()
{
}
// 初始化全部成员变量的构造器
public Address(String addressDetail)
{
this.addressDetail = addressDetail;
}

// addressId的setter和getter方法
public void setAddressId(int addressId)
{
this.addressId = addressId;
}
public int getAddressId()
{
return this.addressId;
}

// addressDetail的setter和getter方法
public void setAddressDetail(String addressDetail)
{
this.addressDetail = addressDetail;
}
public String getAddressDetail()
{
return this.addressDetail;
}

// person的setter和getter方法
public void setPerson(Person person)
{
this.person = person;
}
public Person getPerson()
{
return this.person;
}
}

对于Person的实体来说,也是可控制关联关系的,所以要去掉@OneToMany的注解mappedBy属性,并同时添加@JoinTable注解。

@Entity
@Table(name="person_inf")
public class Person
{
// 标识属性
@Id @Column(name="person_id")
@GeneratedValue(strategy=GenerationType.IDENTITY)
private Integer id;
private String name;
private int age;
// 定义该Person实体所有关联的Address实体
//	@OneToMany(targetEntity=Address.class ,mappedBy="person")
@OneToMany(targetEntity=Address.class)
// 映射连接表,指定连接表为person_address
@JoinTable(name="person_address",
// 指定连接表中person_id列参照当前实体对应数据表的person_id主键列
joinColumns=@JoinColumn(name="person_id"
, referencedColumnName="person_id"),
// 指定连接表中address_id列参照当前实体的关联实体对应数据表的address_id主键列
inverseJoinColumns=@JoinColumn(name="address_id"
, referencedColumnName="address_id", unique=true)
)
private Set<Address> addresses
= new HashSet<>();

// id的setter和getter方法
public void setId(Integer id)
{
this.id = id;
}
public Integer getId()
{
return this.id;
}

// name的setter和getter方法
public void setName(String name)
{
this.name = name;
}
public String getName()
{
return this.name;
}

// age的setter和getter方法
public void setAge(int age)
{
this.age = age;
}
public int getAge()
{
return this.age;
}

// addresses的setter和getter方法
public void setAddresses(Set<Address> addresses)
{
this.addresses = addresses;
}
public Set<Address> getAddresses()
{
return this.addresses;
}
}

保存两个实体的类如下。

public class PersonManager
{
public static void main(String[] args)
{
PersonManager mgr = new PersonManager();
mgr.testPerson();
HibernateUtil.sessionFactory.close();
}

private void testPerson()
{
Session session = HibernateUtil.currentSession();
Transaction tx = session.beginTransaction();
// 创建一个瞬态的Person对象
Person p = new Person();
// 设置Person的name为owen字符串
p.setName("owen");
p.setAge(21);
// 创建一个瞬态的Address对象
Address a = new Address("广州天河");
// 设置Person和Address之间的关联关系
a.setPerson(p);
//创建一个瞬态的Address对象
Address a2 = new Address("上海虹口");
// 设置Person和Address之间的关联关系
p.getAddresses().add(a2);
// 由于采用了连接表来维护1-N关联关系,因此不存在主从表关系,
// 程序可以随意控制先持久化哪个实体。
// 持久化Address对象
session.persist(a2);
// 持久化Person对象
session.save(p);
// 持久化Address对象
session.persist(a);
tx.commit();
HibernateUtil.closeSession();
}
}

执行的SQL语言如下。

Hibernate:
insert
into
address_inf
(addressDetail)
values
(?)
Hibernate:
insert
into
person_inf
(age, name)
values
(?, ?)
Hibernate:
insert
into
address_inf
(addressDetail)
values
(?)
Hibernate:
insert
into
person_address
(person_id, address_id)
values
(?, ?)
Hibernate:
insert
into
person_address
(person_id, address_id)
values
(?, ?)

执行结果如下:

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: