在NHibernate的单表继承模式下通过父类Repository查询子类

简介:

在NHibernate中经常遇到继承与关系数据库的ORMapping的问题,我之前的一篇博客(http://www.cnblogs.com/studyzy/archive/2011/08/16/2140675.html)介绍了有3种常用的实现方式:

  • Concrete Table Inheritance(具体表继承)
  • Single Table Inheritance(单表继承)
  • Class Table Inheritance(类表继承)

其中单表继承是我个人比较常用比较推荐的做法。使用单表继承可以不用Join多个表查询效率高,而且在Domain Model的属性提示到父类或者下降到子类时,数据库模型不用更改。其缺点就是一个数据表的列比较多,而且很多列为空,不过现代的数据库对空列的压缩处理已经比较好了,不会产生大量空列造成的性能问题和磁盘空间的浪费。

在NHibernate中经常会遇到通过父类的Repository来查询子类的情况,比如现在有一个抽象的Employee对象,下面有OfficeUser和Teacher两个具体的对象,这两个对象都有其特有的属性,比如Teacher会有所领导的班级,而OfficeUser却没有。我们可以将OfficeUser和Teacher都保存到Employee表中,然后建立了一个EmployeeRepository,使用Employee作为Query的入口,那么如果要查询Employee表中所有的Teacher该怎么查呢?下面分别用QueryOver、Criteria和HQL来说明:

  • QueryOver查询Employee表中的所有Teacher:
Session.QueryOver<Employee>().Where(a =>a.GetType() == typeof (Teacher)).List();

这里需要注意的是,条件中使用的是a.GetType()==typeof(Teacher),但是不能使用a is Teacher,虽然这在C#中是一回事,但是NHibernate会对其理解不一样,所以必须这么写,否则查询不出来。

  • Criteria查询Employee表中的所有Teacher:
var cri = Session.CreateCriteria<Employee>();
cri=  cri.Add(Expression.Eq("class",typeof(Teacher)));
var ems = cri.List<Employee>();
这里需要注意的是class关键字。必须这么写,不能写成大写的Class,也不能换成其他字,只有这样才能让NHibernate理解,并正确返回结果。
  • HQL查询Employee表中的所有Teacher:
 var cri = Session.CreateQuery("from Employee where Type='Teacher'");  var ems = cri.List<Employee>();

这里又不一样了,where条件后面跟的“Type”是Employee表的Discriminator,是一个列名,也就是通过这一列来区分Employee表的每一条数据到底是OfficeUser还是Teacher。所以,如果你的Discriminator Column是Abc,那么HQL就要写成from Employee where Abc='Teacher'。

以上是以最简单了例子说明了如果通过父类查询具体的子类的方法,实际项目中肯定比这个查询要复杂,但是只要记住了这三种查询的要点,结合其他条件就可以写出NHibernate能够理解的查询。

本文转自深蓝居博客园博客,原文链接:http://www.cnblogs.com/studyzy/archive/2012/07/04/2576902.html,如需转载请自行联系原作者

相关文章
|
2月前
|
缓存 Java 数据库连接
|
5月前
|
Java
Java | 获取实体类中的所有字段,包括继承自其父类的字段
需要通过反射获取一个实体类中的所有字段,包括继承自其父类的字段。
459 0
|
4月前
|
Java 编译器 数据处理
JavaSE——面相对象高级一(4/4)-继承相关的注意事项:权限修饰符、单继承、Object类、方法重写、子类访问成员的特点......
JavaSE——面相对象高级一(4/4)-继承相关的注意事项:权限修饰符、单继承、Object类、方法重写、子类访问成员的特点......
51 0
|
C语言 C++
习题10.3 分别定义Teacher类和Cadre类,采用多重继承方式由这两个类派生出新类Teacher_Cadre类。
习题10.3 分别定义Teacher类和Cadre类,采用多重继承方式由这两个类派生出新类Teacher_Cadre类。
|
Java
Java面向对象(1)--对象的创建使用类的实例化
Java面向对象(1)--对象的创建使用类的实例化
80 1
第五周学习java 继承 在子类父类中有相同参数,子类继承分类后如何进行调用,判断创建的对象属性哪个类
第五周学习java 继承 在子类父类中有相同参数,子类继承分类后如何进行调用,判断创建的对象属性哪个类
第五周学习java 继承 在子类父类中有相同参数,子类继承分类后如何进行调用,判断创建的对象属性哪个类
|
Java
Java面向对象(9)--方法的重写(override/overwrite)
Java面向对象(9)--方法的重写(override/overwrite)
256 0
|
Java
Java - 父类与子类继承调用方法关系
Java - 父类与子类继承调用方法关系
150 0
Java - 父类与子类继承调用方法关系
为什么不通过修改接口或者选择书写继承接口重写方法的新子类,偏偏选择添加上一个装饰器
为什么不通过修改接口或者选择书写继承接口的新子类,偏偏选择添加上一个装饰器?
131 0
为什么不通过修改接口或者选择书写继承接口重写方法的新子类,偏偏选择添加上一个装饰器
|
Java
Java中父类和接口或接口和接口中出现同名属性或同名同参数方法时,如何区分(全文干货)
Java中父类和接口或接口和接口中出现同名属性或同名同参数方法时,如何区分(全文干货)
452 0
下一篇
无影云桌面