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

对于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
        (?, ?)

执行结果如下:

Hibernate的双向1-N关联(五)_第1张图片







你可能感兴趣的:(java,Hibernate,jsp,ssh,Web应用)