用NHibernate处理带属性的多对多关系
1、引言 老谭在面试开发人员的时候,为了考察他们的数据库开发能力,常常祭出我的法宝,就是大学数据库教程中讲到的一个模式:学生选课。这个模式是这样的: 在这个模式中,学生(Student)和课程(Course)都是实体,分别有主键Id。考试成绩(Score)是学生和课程
1、引言
老谭在面试开发人员的时候,为了考察他们的数据库开发能力,常常祭出我的法宝,就是大学数据库教程中讲到的一个模式:学生选课。这个模式是这样的:
在这个模式中,学生(Student)和课程(Course)都是实体,分别有主键Id。考试成绩(Score)是学生和课程之间的多对多关系。
基于这个模式,对于新手,可以出一些简单查询要求,对于熟手,可以出一些复杂的查询要求,用起来得心应手。
但今天要说的重点是,怎么用NHibernate实现这个模式。和一般多对多关系稍有不同的是,这个关系带有一个属性,就是考试成绩(Score)。
如果多对多关系上带有属性,我们一般会把这个关系扩充为实体,也就是把一个多对多关系转换为一个实体加上两个多对一关系。
如果多对多关系上有多个属性,将其转换为一个实体还是有必要的,但只有很少的属性(本例中只有一个),转换为实体实在有些浪费。因为一般情况下,对于实体,我们要为其创建一个人工主键,有了人工主键,又要创建一个序列。在映射时,这个实体自然要有对应的类。这一大堆事情,想想就非常麻烦。
问题的关键是,在概念上这本来就是一个关系,为了实现的方便,而将其转换为一个实体,这个凭空多出来的实体,使概念变得复杂,非常别扭。
因此,这里要探究一下,让关系恢复为为关系,怎么用NHibernate来处理。
2、映射
如果关系表Score中没有Score那个属性字段,Student实体可以映射为这样的类:
public class Student { public virtual long Id { get; set; } public virtual string Name { get; set; } public virtual IList<Course> Courses { get; set; } }
Course也类似。
但有了属性Score,再这样映射就把这个属性丢了。为了带上Score属性,两个实体应该这样映射:
public class Student { public virtual long Id { get; set; } public virtual string Name { get; set; } public virtual IDictionary<Course, int> Courses { get; set; } }
public class Course { public virtual long Id { get; set; } public virtual string Name { get; set; } public virtual IDictionary<Student, int> Students { get; set; } }
原来的列表(List)变成了字典(Dictionary),字典的主键是原来列表中的元素,值则是关系上的属性值,即考试成绩。
对应的映射文件自然也要调整,结果如下:
... <class name="Course" table="Course"> <id name="Id" unsaved-value="0"> <column name="Id" /> ... </id> <property name="Name"> <column name="Name"/> </property> <map name="Students" table="Score" lazy="true"> <key column="CourseId"/> <index-many-to-many column="StudentId" class="Student" /> <element column="Score" /> </map> </class>... <class name="Student" table="Student"> <id name="Id" unsaved-value="0"> <column name="Id" /> ... </id> <property name="Name"> <column name="Name"/> </property> <map name="Courses" table="Score" lazy="true"> <key column="StudentId"/> <index-many-to-many column="CourseId" class="Course"/> <element column="Score" /> </map> </class>
经过这样的映射,多对多关系中的属性带到了类中,而且避免了为关系创建实体——没有Score这样的类。对这样映射结果的操作,和常规多对多关系映射方式多生成的基本类似,但也有几个要注意的问题。
3、查询
对于一些简单的查询,如:
小明所有课程的成绩;化学这门课所有学生的成绩等都比较容易处理,因为这只需要在一个实体上加过滤条件。
如果需要在两个实体上加过滤条件,如查询小明化学课的成绩,就有些复杂,其查询语句Hql是这样的:
select score from Course c, Student s join c.Students score where c.Name='化学' and index(score) = s.Id and s.Name='小明'
这里出现了我们很少用到的hql函数:index()。这是专门应对map的,其目的是获得map的索引字段。
吊诡的是,虽然我们指明,map的key是对象,如Course.Students的key是Student的对象,但index()的结果,仍然是
即便是“简单的查询”,如查询小明所有课程的成绩,也有一个问题要注意。这个问题和懒加载相关。
在这个查询中,我们已经知道需要获取所有课程,因此,希望进行预先加载:
from Student s join fetch s.Courses course where s.Name='小明'
得到结果后,如果脱离查询的环境,如释放Session,在访问课程时,如:
s.Courses.Select(c => c.Key.Name)
目前还不知道怎么预加载map中索引对象。需要的话,只有依赖懒加载机制。
4、维护
除了查询,在对关系进行维护时,也有一点值得特别注意:save-update类型的cascade无效。
cascade属性,为我们进行关系维护带来不少便利。在常规(不带属性)的多对多关系中,我们的维护操作可以是这样的:
小明选化学课:
using (var tx = session.BeginTransaction()) { var student = GetStudent(studentName) ?? new Student { Name = studentName, Courses = new List<Course>() }; var course = GetCourse(courseName) ?? new Course { Name = courseName, Students = new List<Student>() }; if (!course.Students.Contains(student)) { course.Students.Add(student); } session.SaveOrUpdate(course); tx.Commit(); }
其中,GetStudent(studentName) 和 GetCourse(courseName) 分别是指,根据学生名字或课程名字从数据库中加载相应对象。在上面的代码片段中,如果数据库中没有,则新建。维护关系后,进行保存。
需要注意的是,在保存 course 的时候,并没有确保 student 是一个持久化的对象。如果student尚未被持久化,则在保存时,NHibernate会自动保存,并维护和course的关系。能够这么做,依赖于Course中关系上的cascade属性定义(第三行末尾):
<class name="Course" table="Course"> ... <bag name="Students" table="Scoure" lazy="true" cascade="save-update"> <key column="CourseId"/> <many-to-many column="StudentId" class="Student" /> </bag> </class>
但在包含属性的多对多关系上,由于要使用map,就无法进行这样的配置——配置了也不生效。如果数据库中尚未保存该学生,我们不得不首先创建并将其持久化(第7行):
using (var tx = session.BeginTransaction()) { var student = GetStudent(studentName); if (student == null) { student = new Student {Name = studentName, Courses = new Dictionary<Course, int>()}; session.Save(student); } var course = GetCourse(courseName) ?? new Course {Name = courseName, Students = new Dictionary<Student, int>()}; if (course.Students.ContainsKey(student)) { course.Students[student] = score; } else { course.Students.Add(student, score); } session.SaveOrUpdate(course); tx.Commit(); }
否则,就等着NHibernate抛出异常吧。
5、结论
用Map/Dicitionary表达的多对多关系,要比用Bag/List所表达的,操作起来更为复杂。但这样的代价,我们乐意承担。
这是因为,我们更看重模型设计,更重视概念完整性。是模型决定具体实现,而不是反过来,根据具体实现来修改模型的设计。

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

1.首先我們右鍵點選任務列空白處,選擇【任務管理器】選項,或右鍵開始徽標,然後再選擇【任務管理器】選項。 2.在開啟的任務管理器介面,我們點選最右邊的【服務】選項卡。 3.在開啟的【服務】選項卡,點選下方的【開啟服務】選項。 4.在開啟的【服務】窗口,右鍵點選【InternetConnectionSharing(ICS)】服務,然後選擇【屬性】選項。 5.在開啟的屬性窗口,將【開啟方式】修改為【禁用】,點選【應用程式】後點選【確定】。 6.點選開始徽標,然後點選關機按鈕,選擇【重啟】,完成電腦重啟就行了。

在PHP開發過程中,處理特殊字元是常見的問題,尤其是在字串處理中經常會遇到特殊字元轉義的情況。其中,將特殊字元轉換單引號是比較常見的需求,因為在PHP中,單引號是一種常用的字串包裹方式。在本文中,我們將介紹如何在PHP中處理特殊字元轉換單引號,並提供具體的程式碼範例。在PHP中,特殊字元包括但不限於單引號(')、雙引號(")、反斜線()等。在字串

python套件管理器是一個強大且方便的工具,用於管理和安裝Python套件。然而,使用時若不謹慎,可能會陷入各種陷阱。本文將介紹這些陷阱以及應對策略,以幫助開發者避免它們。陷阱1:安裝衝突問題:當多個套件提供具有相同名稱但不同版本的函數或類別時,可能會發生安裝衝突。應對:在安裝前檢查依賴關係,確保套件之間沒有衝突。使用pip的--no-deps選項避免自動安裝依賴項。陷阱2:舊版套件問題:如果未指定版本,套件管理器可能會安裝最新版本,即使有更穩定或適合您需求的舊版本。應對:在安裝時明確指定所需版本,例如p

JavaServerPages(jsP)是一種Java技術,用於建立動態的WEB應用程式。 JSP腳本在伺服器端執行,並在客戶端渲染為html。然而,JSP應用程式容易受到各種安全漏洞的影響,這些漏洞可能導致資料外洩、程式碼執行或拒絕服務。常見安全漏洞1.跨站點腳本(XSS)XSS漏洞允許攻擊者將惡意腳本注入Web應用程序,這些腳本將在受害者訪問頁面時執行。攻擊者可以使用這些腳本竊取敏感資訊(如cookie和會話ID)、重新導向使用者或破壞頁面。 2.注入漏洞注入漏洞允許攻擊者向Web應用程式的資料庫查詢

python物件關聯映射(ORM)是一種技術,它允許Python物件和關係型資料庫表之間進行無縫互動。在人工智慧(ai)和機器學習(ML)應用中,ORM發揮著至關重要的作用,簡化了資料存取和管理,並提高了開發效率。資料儲存和管理ORM提供了一個物件導向的介面來存取和操作資料庫。在AI和ML專案中,通常需要處理大量的數據,包括訓練資料集、模型參數和預測結果。 ORM允許開發人員以簡單易懂的方式與這些資料交互,而無需擔心底層的sql語法。這大大減少了開發時間和錯誤的可能性。例如,在使用Tensorfl

版本控制系統(VCS)是軟體開發中不可或缺的工具,它允許開發人員追蹤和管理程式碼變更。 git是一個流行且功能強大的VCS,廣泛應用於Java開發。本指南將介紹Git的基本概念和操作,為Java開發人員提供版本控制的基礎知識。 Git的基本概念倉庫:程式碼和版本歷史記錄儲存的位置。分支:程式碼庫中的獨立開發線,允許開發人員在不影響主開發線的情況下進行更改。提交:程式碼庫中程式碼的一次更改。回滾:將程式碼庫恢復到先前的提交。合併:將兩個或多個分支中的變更合併到一個分支。 Git入門1.安裝Git從官方網站下載並

隨著分散式系統和多核心處理器的興起,並發集合在現代軟體開發中變得至關重要。 java並發集合提供了高效且線程安全的集合實現,同時管理並發存取的複雜性。本文探討了Java並發集合的未來展望,重點在於新特性和發展趨勢。新特性JSR354:彈性並發集合jsR354定義了具有彈性行為的新並發集合接口,即使在極端並發條件下也能確保性能和可靠性。這些介面提供了原子性的附加功能,例如支援可變不變性和非阻塞迭代。 RxJava3.0:反應式並發集合RxJava3.0引入了反應式程式設計概念,使同時集合能夠與反應式資料流輕

文件是電腦系統中資訊儲存和管理的基本單元,也是Java檔案操作的核心關注點。理解文件的本質對於有效地操作和管理文件至關重要。抽象和層次結構檔案本質上是一個抽象概念,代表著儲存在持久性媒體(例如磁碟或記憶體)中的一組資料。文件的邏輯結構通常由作業系統定義,並提供對資料的組織和存取機制。在Java中,檔案透過File類別表示,它提供了對檔案系統的抽象存取。資料持久性文件的關鍵特性之一是其資料持久性。與記憶體中的資料不同,文件中的資料即使在應用程式退出後仍然存在。這種持久性使得文件成為長期儲存和共享資訊的有
