c# – 如何使用NHibernate加载大型复杂对象图

给定一个像这样的对象图:

A { IEnum<B> }
B { IEnum<C>, IEnum<D>, IEnum<E>, ... }
C { IEnum<X> }

如何在没有N 1问题的情况下急切加载整个对象图?

这是我最终要执行的查询的伪代码:

var a = Session.Get<A>(1); // Query 1
var b_Ids = foreach(b in A.B's) => Select(b.Id); // Query 2
var c = Session.CreateQuery("from C where B in (b_Ids)").Future<C>(); // Query 3
var d = Session.CreateQuery("from D where B in (b_Ids)").Future<D>(); // Query 3
var e = Session.CreateQuery("from E where B in (b_Ids)").Future<E>(); // Query 3

// Iterate through c, d, e, ... find the correct 'B' parent, add to collection manually

我使用这种方法的问题是,当我将’C’,’D’和’E’的实例添加到父’B’的相应集合时,该集合仍然被代理,何时.调用Add(),代理初始化自己并执行更多查询;我认为NHibernate无法看到我已经拥有一级缓存中的所有数据,这是可以理解的.

我尝试通过在Add方法中执行以下操作来解决此问题:

void Add(IEnum<C>)
{
    _collection = new Collection<C>(); // replace the proxied instance to prevent initialization
    foreach(c) => _collection.Add(c);
}

这给了我一个我想要的最佳查询策略,但是后来在做持久性时赶上了我(NHibernate跟踪原始集合的-ref,从我能说的地方).

所以我的问题是,如何在没有N 1的情况下为孩子的孩子加载一个复杂的图表?我到目前为止唯一遇到的就是加入B-C,B-D,B-E,这在我的情况下是不可接受的.

我们使用NH 2.1.2和FluentHN进行映射.升级到NH的v3或使用hbm /存储过程/任何不在桌面之外的东西.

更新:
其中一条评论引用了一种加入方法,我确实遇到了一个演示这种方法的博客.这种解决方法在我们的情况下是不可接受的,但它可能会帮助其他人:Eager fetch multiple child collections in 1 round trip with NHibernate

更新2:
乔丹的回答让我看到了与我的问题相关的以下帖子:Similar QuestionAyende’s blog.此时待定的问题是“如何在没有往返路径的情况下执行子选择”.

更新3:
即使subselect解决方案不是最优的,我也接受了Jordan的回答.

最佳答案 您可以使用可在映射文件中设置的SubSelect提取.这将避免N 1和笛卡尔积.

点赞