目錄
1、引言
2、映射
3、查询
4、维护
5、结论
首頁 資料庫 mysql教程 用NHibernate处理带属性的多对多关系

用NHibernate处理带属性的多对多关系

Jun 07, 2016 pm 04:04 PM
關係 處理 引言

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=&#39;化学&#39;
   and index(score) = s.Id
   and s.Name=&#39;小明&#39;
登入後複製

这里出现了我们很少用到的hql函数:index()。这是专门应对map的,其目的是获得map的索引字段。

吊诡的是,虽然我们指明,map的key是对象,如Course.Students的key是Student的对象,但index()的结果,仍然是 中 column字段的值。在上面的查询中,index(score),我们期望的结果是Student对象,但结果却是对象的Id。因此,在查询语句中,我们不得不关联上Student s,并利用 s.Name 进行过滤。

即便是“简单的查询”,如查询小明所有课程的成绩,也有一个问题要注意。这个问题和懒加载相关。

在这个查询中,我们已经知道需要获取所有课程,因此,希望进行预先加载:

 from Student s
 join fetch s.Courses course
where s.Name=&#39;小明&#39;
登入後複製

得到结果后,如果脱离查询的环境,如释放Session,在访问课程时,如:

s.Courses.Select(c => c.Key.Name)
登入後複製
仍会抛出异常。因为,上述的查询并没有把Course对象加载进来。
目前还不知道怎么预加载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所表达的,操作起来更为复杂。但这样的代价,我们乐意承担。

这是因为,我们更看重模型设计,更重视概念完整性。是模型决定具体实现,而不是反过来,根据具体实现来修改模型的设计。

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

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

AI Clothes Remover

AI Clothes Remover

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

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

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

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

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

熱門話題

Java教學
1665
14
CakePHP 教程
1424
52
Laravel 教程
1322
25
PHP教程
1270
29
C# 教程
1249
24
WIN10服務主機太佔cpu的處理操作過程 WIN10服務主機太佔cpu的處理操作過程 Mar 27, 2024 pm 02:41 PM

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

學習PHP中如何處理特殊字元轉換單引號 學習PHP中如何處理特殊字元轉換單引號 Mar 27, 2024 pm 12:39 PM

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

Python 套件管理器的天坑陷阱:如何避免 Python 套件管理器的天坑陷阱:如何避免 Apr 01, 2024 am 09:21 AM

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

Java JSP 安全漏洞:防護您的 Web 應用程式 Java JSP 安全漏洞:防護您的 Web 應用程式 Mar 18, 2024 am 10:04 AM

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

Python ORM 在人工智慧和機器學習中的作用 Python ORM 在人工智慧和機器學習中的作用 Mar 18, 2024 am 09:10 AM

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

Java Git 入門:版本控制的入門指南 Java Git 入門:版本控制的入門指南 Mar 27, 2024 pm 02:21 PM

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

Java 並發集合的未來展望:探索新特性與發展趨勢 Java 並發集合的未來展望:探索新特性與發展趨勢 Apr 03, 2024 am 09:20 AM

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

Java 檔案操作的哲學:理解文件的本質 Java 檔案操作的哲學:理解文件的本質 Mar 21, 2024 pm 03:20 PM

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

See all articles