作者:ssslinppp
1. 摘要
主要讲解N-1关系表的设计,N:从表,1:主表;
从表(N)中使用外键关联到主表(1),单向的;
场景:
许多人(N)可以住在同一个地方(1),知道人就可以知道这个人的地址;
反过来,知道地址,不能知道该地址住的人。
2. 数据表设计
地址表:


Person表:

e

3. 实体类-注解方式
Address.java

package org.crazyit.app.domain;import javax.persistence.Column;import javax.persistence.Entity;import javax.persistence.GeneratedValue;import javax.persistence.GenerationType;import javax.persistence.Id;import javax.persistence.Table;@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.java

- @JoinColumn :用于定义外键,数据库层面的,name值是数据表中的字段值;
- @ManyToOne :java类层面的;
- @Cascade : 设置级联,系统先自动级联插入主表记录,再插入从表记录==>因为在插入从表(N)记录时,指定的外键不能为null,所以在插入从表记录之前,需要有对应的主表(1)记录存在,若不存在对应的主表记录,则会报错。这里设置@Cascade ,可以保证在主表记录还没有插入而是先插入了从表记录时,Hibernate会自动先将主表记录保存,然后再保存从表记录。这就是级联的意思。
package org.crazyit.app.domain;import javax.persistence.Column;import javax.persistence.Entity;import javax.persistence.GeneratedValue;import javax.persistence.GenerationType;import javax.persistence.Id;import javax.persistence.JoinColumn;import javax.persistence.ManyToOne;import javax.persistence.Table;import org.hibernate.annotations.Cascade;import org.hibernate.annotations.CascadeType;@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;}}
4. 测试

package lee;import org.crazyit.app.domain.Address;import org.crazyit.app.domain.Person;import org.hibernate.Session;import org.hibernate.Transaction;public class PersonManager{public static void main(String[] args){PersonManager mgr = new PersonManager();mgr.testCascase();HibernateUtil.sessionFactory.close();}private void testCascase(){Session session = HibernateUtil.currentSession();Transaction tx = session.beginTransaction();Person p = new Person();Address a = new Address("SuZhou"); // ①p.setName("ZhangSan");p.setAge(21);p.setAddress(a);// 持久化Person对象session.persist(p);// 创建一个瞬态的Address对象Address a2 = new Address("ShangHai"); // ②// 修改持久化状态的Person对象p.setAddress(a2); // ③tx.commit();HibernateUtil.closeSession();}private void testPerson(){Session session = HibernateUtil.currentSession();Transaction tx = session.beginTransaction();// 创建一个Person对象Person p = new Person();// 设置Person的name为crazyit字符串p.setName("WangWu");p.setAge(21);// 持久化Person对象(对应于插入主表记录)session.save(p);// 创建一个瞬态的Address对象Address a = new Address("GuangZhou"); // ①// 先设置Person和Address之间的关联关系p.setAddress(a);// 再持久化Address对象(对应于插入从表记录)session.persist(a);// 创建一个瞬态的Address对象Address a2 = new Address("BeiJing"); // ②// 先设置Person和Address之间的关联关系p.setAddress(a2);// 再持久化Address对象(对应于插入从表记录)session.persist(a2); // ③tx.commit();HibernateUtil.closeSession();}}
执行后的结果:

