Hibernate的单向N-1关联(一)

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

对于无连接的N-1关联而言,程序只要在N的一端增加一列外键,让外键记录对象指所属的实体即可,Hibernate可使用@JoinColumn来修饰代表关联实体的属性,@JoinColumn用于映射底层的外键列。下面定义一个Person的类,Person与Address的关系就是N-1。

@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实体
	@ManyToOne(targetEntity=Address.class)
	// 映射外键列,指定外键列的列名为address_id、不允许为空
	@JoinColumn(name="address_id" , nullable=false)
	@Cascade(CascadeType.ALL)
	private Address address;

	// 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;
	}

	// address的setter和getter方法
	public void setAddress(Address address)
	{
		this.address = address;
	}
	public Address getAddress()
	{
		return this.address;
	}
}

程序无须从Address访问Person,所以Address无须增加Person属性。代码如下。

@Entity
@Table(name="address_inf")
public class Address
{
	// 标识属性
	@Id @Column(name="address_id")
	@GeneratedValue(strategy=GenerationType.IDENTITY)
	private int addressId;
	// 定义地址详细信息的成员变量
	private String addressDetail;
	// 无参数的构造器
	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和Address实体。

private void testCascase()
{
	Session session = HibernateUtil.currentSession();
	Transaction tx = session.beginTransaction();
	// 创建一个Person对象
	Person p = new Person();
	// 创建一个瞬态的Address对象
	Address a = new Address("广州天河");          // ①
	// 设置Person的name为owen.org字符串
	p.setName("owen.org");
	p.setAge(21);
	// 设置Person和Address之间的关联关系
	p.setAddress(a);
	// 持久化Person对象
	session.persist(p);
	// 创建一个瞬态的Address对象
	Address a2 = new Address("上海虹口");         // ②
	// 修改持久化状态的Person对象
	p.setAddress(a2);                             // ③
	tx.commit();
	HibernateUtil.closeSession();
}

上面的实体类中使用了@Cascade(CascadeType.ALL)修饰代表关联实体的属性,这意味着系统先自动级联插入主表记录,也就是先持久化Address对象,再持久化Person对象。也就是说,Hibernate先执行一条insert into address…,再执行一条insert into person….语句。而对于p.setAddress(a2),系统先执行insert into address…,后执行update person….。如果没有@Cascade(CascadeType.ALL),则会报TransientObjectException异常。如下是执行结果图。


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

在有连接的情况下会生成一张中间表,这个时候我们只要修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实体
	@ManyToOne(targetEntity=Address.class)
	// 显式使用@JoinTable映射连接表
	@JoinTable(name="person_address", // 指定连接表的表名为person_address
		// 指定连接表中person_id外键列,参照到当前实体对应表的主键列
		joinColumns=@JoinColumn(name="person_id"
			, referencedColumnName="person_id", unique=true),
		// 指定连接表中address_id外键列,参照到当前实体的关联实体对应表的主键列
		inverseJoinColumns=@JoinColumn(name="address_id"
			, referencedColumnName="address_id")
	)
	private Address address;

	// 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;
	}

	// address的setter和getter方法
	public void setAddress(Address address)
	{
		this.address = address;
	}
	public Address getAddress()
	{
		return this.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字符串
		p.setName("owen");
		p.setAge(21);
		// 创建一个瞬态的Address对象
		Address a = new Address("广州天河");
		// 设置Person和Address之间的关联关系
		p.setAddress(a);
		// 再持久化Address对象
		session.persist(a);
		// 创建一个瞬态的Address对象
		Address a2 = new Address("上海虹口");
		// 设置Person和Address之间的关联关系
		p.setAddress(a2);
		// 由于采用了连接表来维护N-1关联关系,因此不存在主从表关系,
		// 程序可以随意控制先持久化哪个实体。
		// 持久化Address对象
		session.persist(a2);
		// 持久化Person对象
		session.save(p);
		// 持久化Address对象
		session.save(a);
		tx.commit();
		HibernateUtil.closeSession();
	}
}

执行结果如下。






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值