zoukankan      html  css  js  c++  java
  • Hibernate三 关联关系

    Hibernate的关联映射

    客观世界中很少有对象是独立存在的,比如我们可以通过某个老师获取该老师教的所有学生,我们也可以通过某个学生获得教他的对应的老师,实体之间的互相访问就是关联关系。
    在Hibernate中有两种关联关系,即单向关联和双向关联。
    单向关联:只能单向访问关联端,如只能通过老师访问学生。
    双向关联:关联的两端可以互相访问,如老师可以访问学生,学生可以访问老师。
    单向关联可以分为:1->1 1->N N->1 N-N
    双向关联可以分为:1-1 1-N N-N

     
    一 单向关联部分

    1.单向N->1关联
    单向N->1是指可以通过N中的某一个找到1中的实体,所以控制的那一端在N端,使用的是@ManyToOne注解。
    单向N->1关系,比如多个人对应一个地址,可以从人的这一端找到对应的地址实体,不需要关心某个地址的用户。
    无连接表的N->1关联
    Person.java

    @Entity
    @Table(name="person_info")
    public class Person { 
    @Id
    @Column(name="person_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Integer id;
    private String name;
    private int age;
    
    @ManyToOne(targetEntity=Address.class)
    @JoinColumn(name="address_id",nullable=false)
    @Cascade(CascadeType.ALL)
    private Address address;
    ...//此处省略getter/setter等其他方法
    }

    Address.java

    @Entity
    @Table(name="address_info")
    public class Address {
    @Id
    @Column(name="address_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private int addressId;
    private String adressDetail;
    ...//此处省略getter/setter/构造器等方法
    }

    (1)@ManyToOne注解:
    对于无连接表的N->1关联而言,只需要在N的那一端添加@ManyToOne注解,1的那一端作为一个一般的实体类就可。我们用targetEntity属性指定关联实体的类名,大部分时候无需指定targetEntity,但是如果使用@OneToMany或@ManyToMany修饰的1--N、N--N关联,使用Set集合不带泛型信息,就必须指定targetEntity属性
    (2)@JoinColumn注解:
    @JoinColumn表示通过外键关联策略进行映射,也就是说,程序在N的这一端增加一列外键,让外键记录该对象所属的实体,@JoinColumn用于映射底层的外键列。
    (3)cascade注解:指定Hibernate对关联实体采用怎样的级联策略,包括以下五种情况:
    CascadeType.ALL:将所有的持久化操作都级联到关联实体
    CascadeType.MERGE:将merge操作都级联到关联实体
    CascadeType.PERSIST:将persist操作都级联到关联实体
    CascadeType.REFRESH:将refresh操作都级联到关联实体
    CascadeType.REMOVE:将所remove操作都级联到关联实体
    (4)fetch:指定抓取关联实体时的抓取策略
    FetchType.EAGER:立即抓取
    FetchType.LAZY:延迟抓取
    配置文件hibernate.cxf.xml加上:

    <hibernate-configuration>
    <session-factory>
    <property name="hibernate.dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
    <property name="hibernate.connection.driver_class">com.mysql.jdbc.Driver</property>
    <property name="hibernate.connection.url">jdbc:mysql:///mydb</property>
    <property name="hibernate.connection.username">root</property>
    <property name="hibernate.connection.password">123456</property>
    <property name="hibernate.hbm2ddl.auto">update</property>
    <property name="show_sql">true</property>
    <property name="hibernate.format_sql">false</property>
    <mapping class="myHibernate.Person"/>
    <mapping class="myHibernate.Address"/>
    </session-factory>
    </hibernate-configuration>

    测试:

    public class myTest {
        public static void main(String[] args) {
            Configuration config=new Configuration().configure();
            SessionFactory factory=config.buildSessionFactory();
            Session session=factory.openSession();
            Transaction tx=session.beginTransaction();
            Person p=new Person();
            p.setName("lyy");p.setAge(22);
            //创建一个瞬态的Address对象
            Address a=new Address("北京"); 
            //session.save(a);
            p.setAddress(a);
            session.save(p);
            //再次创建一个瞬态的Address对象
            Address a2=new Address("南京"); 
            p.setAddress(a2);
            tx.commit();
            session.close();
        }
    }                    

    程序执行到创建一个瞬态的Address对象时,我们知道,当某一个表中有关联表时,通常是要先保存(删除)关联表Address,然后再保存(删除)从表Person。也就是说,先有session.save(a);然后再有 session.save(p);但是如果我们把session.save(a);去掉,程序会有什么后果呢。这个时候,由于关联表Address还处于瞬态,没有保存,通常情况下会有两种情况出现:
    (1)程序抛出异常:TransientObjectException:object references an unsaved transient instance - save the transient instance before flushing
    (2)系统自动级联插入主表记录,再插入从表记录。
    在上个例子中,程序会出现第二种情况,因为我们指定了@Cascade(CascadeType.ALL)。这意味着系统将会自动级联插入主表记录,即先持久化Address对象,再持久化Person对象。也就是说,Hibernate先执行了一条insert into address...语句,再执行一条insert into person...语句。也就是说,当实体类中没有@Cascade(CascadeType.ALL)这个注解,程序就会报(1)中的异常。
    当程序再次创建一个瞬态的Address对象时,程序会将该瞬态的Address对象关联到已经持久化的Person对象中,注意,这里的Person对象已经save过,是一个持久化的对象,不需要再保存了,重新关联了新的Address对象就相当于是执行了一次更新操作。也就是说,hibernate先执行了一条insert into address...语句,又执行了update person...语句。此时数据库中address表有两列数据,person表有一列数据。

    有连接表的N->1关联
    通常情况下,我们都是使用基于外键的关联映射,很少用有连接表的关联映射,所以,只在N->1关联中举一个例子。
    将Person.java修改为:

    @Entity
    @Table(name="person_info")
    public class Person {
    @Id
    @Column(name="person_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Integer id;
    private String name;
    private int age;
    
    @ManyToOne(targetEntity=Address.class)
    @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"))
    @Cascade(CascadeType.ALL)
    private Address address;
    ...
    }

    joinColumns:可接受多个@joinColumn,用于配置连接表中外键列的列信息,这些外键列参照当前实体对应表的主键列
    inverseJoinColumns:可接受多个@joinColumn,用于配置连接表中外键列的列信息,这些外键列参照当前实体的关联实体对应表的主键列

    2.单向1->1关联
    基于外键的单向1->1关联
    只需要在控制的那个实体中使用@OneToOne注解,使用@JoinColumn映射外键列即可。由于是1->1关联,因此还需要为@JoinColumn增加unique=true即可。
    person.java

    @Entity
    @Table(name="person_info")
    public class Person {
    @Id
    @Column(name="person_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Integer id;
    private String name;
    private int age;
    
    @OneToOne(targetEntity=Address.class)
    @JoinColumn(name="address_id",referencedColumnName="address_id",unique=true)
    @Cascade(CascadeType.ALL)
    private Address address;
    ...
    }

    3.单向1->N关联
    持久化类发生了改变,因为1的一端要访问N的一端,也就是说,通过一个person找到所有的address,这里的address就需要使用集合属性。也就是说,只需要在1的那一端增加Set类型的成员变量。
    person.java

    @Entity
    @Table(name="person_info")
    public class Person {
    @Id
    @Column(name="person_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Integer id;
    private String name;
    private int age;
    
    @OneToMany(targetEntity=Address.class)
    @JoinColumn(name="person_id",referencedColumnName="person_id")
    private Set<Address> addresses=new HashSet<>();
    ...
    }

    注意,使用@JoinColumn映射外键列,不同的是,此处的外键列并不是增加到当前实体对应的数据表中,而是增加到关联实体Adress对应的数据表中。比如说,在单向N->1关联中,person表有id字段,name字段,age字段,还有address_id字段进行外键关联。id不同的person可能有同一个address_id。而在1->N关联中,person表并没有address_id字段,而在address表中,有person_id字段进行关联。其实很好理解,id不同的address可能有同一个person_id。如果将外键关联加到person表中,会发现很有很多个相同的person_id,主键不唯一。
    测试:

    public class myTest {
    public static void main(String[] args) {
    Configuration config=new Configuration().configure();
    SessionFactory factory=config.buildSessionFactory();
    Session session=factory.openSession();
    Transaction tx=session.beginTransaction();
    Person p=new Person();
    Address a=new Address("北京"); 
    session.persist(a);
    p.setName("lyy");p.setAge(22);
    p.getAddresses().add(a);
    session.save(p);
    Address a2=new Address("南京");
    session.persist(a2);
    p.getAddresses().add(a2);
    tx.commit();
    session.close();
    }
    }

    分析一下执行过程,我们就会发现,由于先自动保存关联实体address,而此时address实体中的person_id是无值的,也就是说,程序先通过insert语句插入了一条外键为null的address记录,然后再执行update语句更新刚刚的插入的address记录,这肯定会影响系统性能。应该尽量少用单向1->N关联,而是改为双向1-N关联。而对于双向的1-N关联,使用1的一端控制关联关系会有很多弊端,比如插入数据时无法插入外键列,会额外多出一条update语句,并且外键列还不能增加非空约束。

    4.单向N->N关联
    控制的一端需要添加一个Set类型的属性,被关联的持久化实例以集合形式存在。N-N关联必须使用关联表。

    person.java

    @Entity
    @Table(name="person_info")
    public class Person {
    @Id
    @Column(name="person_id")
    @GeneratedValue(strategy=GenerationType.AUTO)
    private Integer id;
    private String name;
    private int age;
    
    @ManyToMany(targetEntity=Address.class)
    @JoinTable(name="person_address",
    joinColumns=@JoinColumn(name="person_id",referencedColumnName="person_id"),
    inverseJoinColumns=@JoinColumn(name="address_id",referencedColumnName="address_id"))
    private Set<Address> addresses=new HashSet<>();
    ...
    }

    二 双向关联部分

    1.双向1-N关联
    对于1-N关联,Hibernate推荐使用双向1-N关联,而且用N的一端控制关联关系,使用无连接表的映射策略即可。
    N的一端需要增加@ManyToOne注解,并用@JoinColumn映射外键列,而在1的一端使用@OneToMany注解,并指定mappedBy属性,一旦指定了该属性,就说明当前实体不能控制关联关系。
    注意,对于指定了mappedBy的@OneToMany,@ManyToMany,@OneToOne注解,都不能与@JoinColumn或@JoinTable同时修饰代表关联实体的属性。

    Person.java

    @Entity
    @Table(name="person_info")
    public class Person {
        @Id
        @Column(name="person_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private Integer id;
        private String name;
        private int age;
    
        @OneToMany(targetEntity=Address.class,mappedBy="person")
        private Set<Address> addresses=new HashSet<>();
        ...
    }

    Address.java

    @Entity
    @Table(name="address_info")
    public class Address {
        @Id
        @Column(name="address_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private int addressId;
        private String adressDetail;
        @ManyToOne(targetEntity=Person.class)
        @JoinColumn(name="person_id",referencedColumnName="person_id",nullable=false)
        //这里与单向1->N不同,person_id不可以为null
        private Person person;
        ...
    }

    测试:

    public class myTest {
        public static void main(String[] args) {
            Configuration config=new Configuration().configure();
            SessionFactory factory=config.buildSessionFactory();
            Session session=factory.openSession();
            Transaction tx=session.beginTransaction();
            Person p=new Person();
            p.setName("lyy");p.setAge(22);
            session.save(p);   //持久化person对象
            Address a=new Address("北京");
            a.setPerson(p);    //设置person和address之间的关联关系
            session.persist(a); //持久化address对象
            Address a2=new Address("南京");
            a2.setPerson(p);
            session.persist(a2);
            tx.commit();
            session.close();
        }
    }

    由程序可以发现,最好先持久化person对象,因为程序希望在持久化address对象时,可以为address的外键列person_id分配值。

    2.双向N-N关联
    双向N-N关联需要在两端都使用Set集合属性,两端都增加对集合属性的访问,只能采用连接表来建立两个实体之间的关联关系。
    也就是说,两端都要使用@ManyToMany修饰Set集合属性,并在两端都使用@JoinTable显示映射连接表,并且两端指定的连接表的表名应该相同,指定的外键列的列名相互对应。
    如果程序希望某一端放弃控制关联关系,则可以在这一端的@ManyToMany注解中指定MappedBy属性,这一端也就不能再指定@JoinTable映射连接表了。
    Person.java

    @Entity
    @Table(name="person_info")
    public class Person {
        @Id
        @Column(name="person_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private Integer id;
        private String name;
        private int age;
    
        @OneToMany(targetEntity=Address.class)
        @JoinTable(name="person_address",
        joinColumns=@JoinColumn(name="person_id",referencedColumnName="person_id"),
        inverseJoinColumns=@JoinColumn(name="address_id",referencedColumnName="address_id",unique=true))
        private Set<Address> addresses=new HashSet<>();
        ...
    }

    Address.java

    @Entity
    @Table(name="address_info")
    public class Address {
        @Id
        @Column(name="address_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private int addressId;
        private String adressDetail;
        @ManyToOne(targetEntity=Person.class)
        @JoinTable(name="person_address",
        joinColumns=@JoinColumn(name="address_id",referencedColumnName="address_id"),
        inverseJoinColumns=@JoinColumn(name="person_id",referencedColumnName="person_id"))
        private Person person;
        ...
    }

    注意,由于此处管理的是N-N关联,就不能为@JoinColumn注解增加unique=true。

    3.双向1-1关联
    两端都要使用@OneToOne注解进行映射,外键可以存放在任意一端,即通过@JoinColumn注解来映射外键列。一旦选择其中的一端来增加外键,该表即变为从表,另一个表则为主表。
    双向1--1关联的主表对应的实体,也不应该用于控制关联关系,因此要用mappedBy属性
    Person.java

    @Entity
    @Table(name="person_info")
    public class Person {
        @Id
        @Column(name="person_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private Integer id;
        private String name;
        private int age;
    
        @OneToOne(targetEntity=Address.class,mappedBy="person")
        private Address address;
        ...
    }

    Address.java

    @Entity
    @Table(name="address_info")
    public class Address {
        @Id
        @Column(name="address_id")
        @GeneratedValue(strategy=GenerationType.AUTO)
        private int addressId;
        private String adressDetail;
        @OneToOne(targetEntity=Person.class)
        @JoinColumn(name="person_id",referencedColumnName="person_id",unique=true)
         private Person 
    }
  • 相关阅读:
    HDU 5441 离线处理 + 并查集
    [转载]HDU 3478 判断奇环
    POJ 1637 混合图的欧拉回路判定
    [转载] 一些图论、网络流入门题总结、汇总
    UVA 820 --- POJ 1273 最大流
    [转载 ]POJ 1273 最大流模板
    POJ 3041 -- 二分图匹配
    2014西安现场赛F题 UVALA 7040
    UVA 12549
    割点、桥(一点点更新)
  • 原文地址:https://www.cnblogs.com/lyy-2016/p/5726916.html
Copyright © 2011-2022 走看看