代码改变世界

使用NHibernate(8)-- 延迟加载

2014-04-08 15:43  FuzhePan  阅读(334)  评论(0编辑  收藏  举报

1,延迟加载。

延迟加载,即用到的时候再加载数据。这种机制是非常有情怀的,比如一篇中的用户实体有标签、问题等导航属性,如果只是用到用户名去查询整个实体,则把相关的标签和问题也都加载,性能会比较低。而有了延迟加载机制之后就可以做到查询用户实体只加载用户数据,访问到标签或问题等属性的时候再去加载这些数据。

如果使用Linq To NHibernate,则可以使用Linq提供的延迟加载机制,对于这种,是linq提供的机制,就不讨论了。

需要注意的是,如果查询出用户实体之后,关闭了Session。则延迟加载的时候会抛出异常。这是因为关闭session数据库连接被关闭,无法继续查询数据。

NHibernate默认是开启延迟加载的,也可以通过Lazy="false"属性关闭某个属性的延迟加载机制,这时会立即加载。

可以通过查看生成的sql来验证这种情况,再此就不写示例代码了。

2,延迟加载的原理。

延迟加载一般是通过代理模式来实现的,NHibernate会为我们的实体生成一个继承与该实体的代理,并重写父类的方法(这也是NHibernate要求我们的实体属性都必须是Virtual的原因)。

比如对于我们的User类,会生类似如下的代理类:

public class UserProxy:User
{

    public override List<Question> Questions

    {

        get

        {

            base.Questions = ...//加载数据

            return base.Questions;

        }

        set

        {

            base.Questions = value;

        }

    }

}

因为继承的关系,所以把代理类赋值给User类是没有问题的,同时由于多态,调用User的Question的时候实际调用的是代理类的属性。

3,方法Get和Load的区别。

Session有两个查询数据的方法,及Get和Load。Get方法会立即从数据库加载对象,而Load方法返回的是一个代理对象,当使用这个代理对象时,再去查询数据库,从而实现延迟加载。

这一点也可以通过监视生成的sql来实现。

4,Linq To NHibernate中的立即查询。

有时候,我们明确知道在某个业务逻辑中会用到当前实体的某个导航属性,希望查询的时候一次查询出来提高效率,而不是先加载实体,再加载导航属性。这时候可以用Linq To NHibernate中的立即查询来实现。

NHibernate提供了四种方法:Fetch及ThenFetch,FetchMany及ThenFetchMany。

Fetch用来加载关联关系,加载User的同时加载导航属性Question集合:

var users = session.Query<User>().Fetch(u=>u.Questions).ToList();

如果Question实体又包含Answer的导航属性,希望也一并加载出来,则可以使用ThenFetch:

var users = session.Query<User>().FetchMany(u=>u.Questions).ThenFetch(q=>q.Answers).ToList();

对于Fetch和FetchMany的区别:

针对上面的代码Fetch返回的是Question集合,不能针对返回结果继续ThenFetch。

而FetchMany返回的是Question,可以对其结果继续进行ThenFetch。