Hibernate(或其它ORM)里的inverse用法详解,内容摘自Java web轻量级开发面试教程

Java基础

浏览数:51

2019-6-3

    本文来是从 java web轻量级开发面试教程从摘录的。

    Inverse的英文含义是反转,在Hibernate中用来决定是由哪方来维护两个业务实体类之间的关联关系,具体而言,就是由哪方去设置这个被外键约束的字段值。

    它的默认值是false,也就是说,本端(比如inverse=false写在学生端,那么本端是学生,另外一方是课程)不“反转控制权”,这句别扭的话的另外一种说法是,本端维护关联关系。如果两边都不写,那么两端都维护。这样会造成问题,即新时因为两端都控制关系,因此可能会导致重复更新。

    注意,inverse仅仅是指定由谁来设置外键值,而不是用来设置级联操作,级联操作的方式由cascade来负责,很多人会混淆它们的含义和用法。

   这个例子所用到的数据表是Person表和Card表,其中Person表里包含ID、Name和Phone字段,而Card表里包含了CardID、PersonID、Bank和balance四个字段。   

    在hibernate.cfg.xml文件里,通过mapping resource来指定对应的映射文件,其中关键的代码如下:   

1	<!-- 添加实体类的映射文件-->
2	<mapping resource="Model/Card.hbm.xml" />
3	<mapping resource="Model/Person.hbm.xml" />

    在Person.hbm.xml里,用如下代码来指定人(一方)和卡(多方)的关系,其中在人这一端,inverse是true。   

1	<set name="cards" cascade="save-update" inverse="true">
2	<key column="PersonID"/>
3	<one-to-many class="Card"/>
4	</set>

    在HibernateMain.java里,通过如下关键代码插入了一个人的信息。    

1	//创建一张卡
2	Card card1 = new Card();
3	        card1.setCardID("Card123");
4	        card1.setPersonID("Person123");
5	        card1.setBank("Citi");
6	        card1.setBalance(100f);
7	//初始化一个人
8	        Person person = new Person();
9	        person.setID("Person123");
10	        person.setName("Peter");
11	        person.setPhone("123456");
12	//在cards这个set里加入card1
13	        Set<Card> cards = new HashSet<Card>();
14	        cards.add(card1);
15	        person.setCards(cards);
16	//保存人的信息
17	        session.save(person);
18	        session.flush();

    运行后,在输出信息里能看到如下两条相关的插入语句:一条是插入Person信息,另一条是插入card信息。

1     Hibernate: insert into Person (Name, Phone, ID) values (?, ?, ?)

2     Hibernate: insert into Card (PersonID, Bank, Balance, Person, CardID) values (?, ?, ?, ?, ?)

    当把person.hbm.xml里的inverse设置成false时,能看到相关的语句里会多出一句update语句。

1     Hibernate: insert into Person (Name, Phone, ID) values (?, ?, ?)

2     Hibernate: insert into Card (PersonID, Bank, Balance, Person, CardID) values (?, ?, ?, ?, ?)

3     Hibernate: update Card set PersonID=? where CardID=?

    原因是,当设置inverse为true时,Person这一端反转外键控制权,也就是由Card这端来管理外键,而在代码里我们仅仅是插入了Person,没有插入Card,所以就没有更新两个外键(PersonID和CardID)的操作。相反,当inverse为false时,管理外键控制权是在Person端,那么当插入Person时,Hibernate就需要额外的一句update语句来管理外键了。

    在一对多的例子里,inverse不论取什么值,对结果都没有影响,所以很容易让人忽视它的作用。

    在一对多的例子里,一般是让多方来管理外键控制权,比如一个人有100张开,那么如果由Person方来管理的话,无形中可能会多出100个update操作,效率上就不大好了。

    如果在一对多案例中,inverse只是影响效率的话,那么在多对多的例子中,inverse的设置就可能影响到数据。

    我们再来看个多对多的学生选课案例,其中一个学生可以选多门课,而一门课里可以有多个学生。在Student.hbm.xml里,描述多对多关系的语句里可加上inverse=“true”的语句。   

1	<set name="courses" table="students_courses" inverse= "true" cascade="save-update">
2	<key column="student_id"></key>
3	<many-to-many class="Model.Course" column="course_id"></many-to-many>
4	</set>

    在Course.hbm.xml里,不加任何关于inverse的语句,也就是说,在Student端反转外键控制权,把控制权交到Course端。

    在HibernateMain这个类里,通过如下代码让s1学生选修计算机课程。     

1	        //设置一个学生,学号是1
2	Student s1 = new Student();
3	        s1.setStudentID("1");
4	        s1.setStudentName("Peter");
5	//设置多个课程
6	        Course c1 = new Course();
7	        c1.setCourseID("c1");
8	        c1.setCourseName("Math");
9	        Course c2 = new Course();
10	        c2.setCourseID("c2");
11	        c2.setCourseName("Java");
12	        Course c3 = new Course();
13	        c3.setCourseID("c3");
14	        c3.setCourseName("C#");
15	//设置计算机课程这个Set
16	        Set<Course> computerCourses = new HashSet<Course>();
17	        computerCourses.add(c2);
18	        computerCourses.add(c3);
19	//让s1这个学生选修计算机课程(也就是c2和c3课程)
20	        s1.setCourses(computerCourses);
21	//保存s1
22	        session.save(s1);
23	        session.flush();

    执行结果是,虽然能在Student和Course表里看到相关的学生和课程的记录,但在关键的描述学生选课关联表(students_courses)里,看不到任何关联记录。原因是已经通过设置inverse把外键管理权交给Course方了,这里仅仅是保存学生,并没有保存课程,所以没有插入外键的动作。如果要在students_courses表里插入外键关联,就需要在person.hbm.xml里设置inverse的值为false。

    所以,在多对多关联里,设置错了inverse值会导致结果出错,请大家根据具体项目的情况适当设值。

      

  

  

    

 

作者:hsm_computer