java Hibernate多對(duì)多映射詳解及實(shí)例代碼
java Hibernate多對(duì)多映射
前言:
一、單向多對(duì)多
單向多對(duì)多的例子用人和職位來(lái)舉例,一個(gè)人可以有多個(gè)職位,一個(gè)職位會(huì)有多個(gè)人。單向多對(duì)多是指只能在一端來(lái)查詢(xún)獲取另一端的內(nèi)容。多對(duì)多的關(guān)系在生成關(guān)系模型時(shí)會(huì)生成對(duì)象之前的關(guān)聯(lián)表,關(guān)聯(lián)表中存放著兩個(gè)關(guān)系表的主鍵,它們的關(guān)系如下所示:
代碼部分:
(1)映射和關(guān)系類(lèi)
因?yàn)槭菃蜗虻年P(guān)系,所以只需要在一端進(jìn)行維護(hù),所以我們需要在User.hbm.xml配置文件中添加<many-to-many>標(biāo)簽,并在標(biāo)簽中加上對(duì)應(yīng)的列關(guān)系,在<set>表中添加table屬性來(lái)指明生成新表,User.hbm.xml代碼如下:
<span style="font-family:KaiTi_GB2312;font-size:18px;"><hibernate-mapping> <class name="com.bjpowernode.hibernate.User" table="t_user"> <id name="id"> <generator class="native"/> </id> <property name="name"/> <set name="roles" table="t_user_role"> <key column="user_id"/> <many-to-many class="com.bjpowernode.hibernate.Role" column="role_id" /> </set> </class> </hibernate-mapping></span>
Role.hbm.xml代碼比較簡(jiǎn)單,不需要添加多余的標(biāo)簽來(lái)維護(hù)關(guān)系:
<hibernate-mapping> <class name="com.bjpowernode.hibernate.Role" table="t_role"> <id name="id"> <generator class="native"/> </id> <property name="name"/> </class> </hibernate-mapping>
因?yàn)閡ser的映射中有set映射,所以需要在相應(yīng)的類(lèi)文件中添加Hashset,User.java代碼如下:
<span style="font-family:KaiTi_GB2312;font-size:18px;">import java.util.Set; public class User { private int id; private String name; private Set roles; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Set getRoles() { return roles; } public void setRoles(Set roles) { this.roles = roles; } }</span>
Role.java代碼如下:
public class Role { private int id; private String name; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } }
(2)添加和讀取數(shù)據(jù):
進(jìn)行添加數(shù)據(jù)時(shí),需要首先吧關(guān)系保存到數(shù)據(jù)庫(kù)中,然后創(chuàng)建用戶(hù)Hash表,在hash表中添加對(duì)應(yīng)的關(guān)系,最后創(chuàng)建用戶(hù),將hash表添加到用戶(hù)上。這部分需注意的是寫(xiě)入的先后順序,否則會(huì)出現(xiàn)很多null值,和之前的映射一樣的道理。
public void testSave1() { Session session = null; try { session = HibernateUtils.getSession(); session.beginTransaction(); Role r1 = new Role(); r1.setName("數(shù)據(jù)錄入人員"); session.save(r1); Role r2 = new Role(); r2.setName("商務(wù)主管"); session.save(r2); Role r3 = new Role(); r3.setName("商務(wù)經(jīng)理"); session.save(r3); Role r4 = new Role(); r4.setName("項(xiàng)目會(huì)計(jì)"); session.save(r4); User u1 = new User(); u1.setName("張三"); Set u1Roles = new HashSet(); u1Roles.add(r1); u1Roles.add(r2); u1.setRoles(u1Roles); session.save(u1); User u2 = new User(); u2.setName("李四"); Set u2Roles = new HashSet(); u2Roles.add(r1); u2Roles.add(r2); u2Roles.add(r3); u2.setRoles(u2Roles); session.save(u2); User u3 = new User(); u3.setName("王五"); Set u3Roles = new HashSet(); u3Roles.add(r3); u3Roles.add(r4); u3.setRoles(u3Roles); session.save(u3); session.getTransaction().commit(); }catch(Exception e) { e.printStackTrace(); session.getTransaction().rollback(); }finally { HibernateUtils.closeSession(session); } }
讀取時(shí)因?yàn)槭菃蜗蜿P(guān)系,只需要通過(guò)一來(lái)讀取另一端的內(nèi)容,通過(guò)user來(lái)讀取role的內(nèi)容。代碼如下:
public void testLoad1() { Session session = null; try { session = HibernateUtils.getSession(); session.beginTransaction(); User user = (User)session.load(User.class, 2); System.out.println(user.getName()); for (Iterator iter=user.getRoles().iterator(); iter.hasNext();) { Role role = (Role)iter.next(); System.out.println(role.getName()); } session.getTransaction().commit(); }catch(Exception e) { e.printStackTrace(); session.getTransaction().rollback(); }finally { HibernateUtils.closeSession(session); } }
二、雙向多對(duì)多映射
和之前介紹的一樣,雙向多對(duì)多就是在兩端同時(shí)維護(hù)關(guān)系,從任何一端都能加載到另一端的內(nèi)容,話(huà)不多說(shuō)直接上代碼:
因?yàn)槭请p向的所以需要同時(shí)加入雙向的集合映射,在配置文件中添加<set>標(biāo)簽,添加多對(duì)多標(biāo)簽,Role.hbm.xml代碼如下:
<hibernate-mapping> <class name="com.bjpowernode.hibernate.Role" table="t_role"> <id name="id"> <generator class="native"/> </id> <property name="name"/> <set name="users" table="t_user_role"> <key column="role_id" not-null="true"/> <many-to-many class="com.bjpowernode.hibernate.User" column="user_id"/> </set> </class> </hibernate-mapping>
User.hbm.xml代碼如下,和單向映射代碼是一樣的:
<span style="font-family:KaiTi_GB2312;font-size:18px;"><hibernate-mapping> <class name="com.bjpowernode.hibernate.Role" table="t_role"> <id name="id"> <generator class="native"/> </id> <property name="name"/> <set name="users" table="t_user_role"> <key column="role_id" not-null="true"/> <many-to-many class="com.bjpowernode.hibernate.User" column="user_id"/> </set> </class> </hibernate-mapping> </span>
Role.java部分,和單向的user.java一樣需要添加集合映射set,代碼如下:
import java.util.Set; public class Role { private int id; private String name; private Set users; public int getId() { return id; } public void setId(int id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public Set getUsers() { return users; } public void setUsers(Set users) { this.users = users; } }
User.hbm.xml和User.java代碼和上文中的代碼相同,就不全部放上來(lái)了。
小結(jié):
單向和多向通過(guò)幾篇博客的介紹相信大家已經(jīng)明白,我們只需要記住單向的雙向的也就會(huì)了,挺簡(jiǎn)單的。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
SpringBoot?2.5.5整合輕量級(jí)的分布式日志標(biāo)記追蹤神器TLog的詳細(xì)過(guò)程
分布式追蹤系統(tǒng)是一個(gè)最終的解決方案,如果您的公司已經(jīng)上了分布式追蹤系統(tǒng),這篇文章主要介紹了SpringBoot?2.5.5整合輕量級(jí)的分布式日志標(biāo)記追蹤神器TLog,需要的朋友可以參考下2022-10-10Java讀寫(xiě)pdf文件的詳細(xì)實(shí)現(xiàn)方法
最近公司的項(xiàng)目中需要操作pdf文件,所以這里給大家總結(jié)下方法,這篇文章主要給大家介紹了關(guān)于Java讀寫(xiě)pdf文件的詳細(xì)實(shí)現(xiàn)方法,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-09-09java中Pulsar?InterruptedException?異常
這篇文章主要為大家介紹了java中Pulsar?InterruptedException?異常分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02mybatis-plus通用枚舉@JsonValue接收參數(shù)報(bào)錯(cuò)No enum constant
最近在使用mybatis-plus時(shí)用到了通用枚舉,遇到了問(wèn)題,本文主要介紹了mybatis-plus通用枚舉@JsonValue接收參數(shù)報(bào)錯(cuò)No enum constant,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09一文詳解Springboot中filter的原理與注冊(cè)
這篇文章主要為大家詳細(xì)介紹了Springboot中filter的原理與注冊(cè)的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),對(duì)我們掌握SpringBoot有一定的幫助,需要的可以參考一下2023-02-02Java基于jdbc實(shí)現(xiàn)的增刪改查操作示例
這篇文章主要介紹了Java基于jdbc實(shí)現(xiàn)的增刪改查操作,結(jié)合實(shí)例形式分析了java使用jdbc進(jìn)行數(shù)據(jù)庫(kù)的連接、增刪改查等基本操作技巧,需要的朋友可以參考下2019-01-01java實(shí)現(xiàn)上傳和下載工具類(lèi)
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)上傳和下載工具類(lèi),文件上傳到ftp服務(wù)工具類(lèi),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-05-05Junit單元測(cè)試關(guān)于@Transactional注解引起的事務(wù)回滾問(wèn)題
這篇文章主要介紹了Junit單元測(cè)試關(guān)于@Transactional注解引起的事務(wù)回滾問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08