1

我正在尝试在 NHibernate 中使用全局过滤器,据我所知,我正在做所有教程所做的事情,但我遇到了一个例外。

我的 .hbm.xml 文件包含以下内容:

...
<class name="NHibernateSandbox.Foo, NHibernateSandbox" table="Foo">
    ...
    <property column="Content" type="String" name="Content" not-null="true" length="100" />
    <property column="Deleted" type="Boolean" name="Deleted" not-null="true" />
    <filter name="Foo_Nondeleted" condition="Deleted = false" />
</class>

然后我有一个简单的测试类:

Configuration cfg = new Configuration();
cfg.Configure();

using (ISessionFactory sf = cfg.BuildSessionFactory()) {
    using (ISession session = sf.OpenSession()) {
        session.EnableFilter("Foo_Nondeleted");
        IQuery query = session.CreateQuery("FROM NHibernateSandbox.Foo");
        foreach (Foo foo in query.List<Foo>()) {
            Console.WriteLine(foo.Content);
        }
    }
}

如果我删除该EnableFilter行,它会按预期工作:打印已删除和未删除的行。但是,通过这EnableFilter条线,我得到了 NHibernateException

No such filter configured [Foo_Nondeleted]
  at NHibernate.Impl.SessionFactoryImpl.GetFilterDefinition(String filterName)
  at NHibernate.Impl.SessionImpl.EnableFilter(String filterName)
  at NHibernateSandbox.Program.Main(String[] args)

如果我将 log4net 配置为详细,那么我会看到

INFO  NHibernate.Cfg.HbmBinder - Mapping class: NHibernateSandbox.Foo -> Foo
DEBUG NHibernate.Cfg.HbmBinder - Mapped property: Id -> RID, type: Int32
DEBUG NHibernate.Cfg.HbmBinder - Mapped property: Content -> Content, type: String
DEBUG NHibernate.Cfg.HbmBinder - Mapped property: Deleted -> Deleted, type: Boolean
DEBUG NHibernate.Cfg.HbmBinder - Applying filter [Foo_Nondeleted] as [Deleted = false]

“应用过滤器”和“配置”过滤器并可供会话使用之间缺少什么步骤?

4

1 回答 1

5

向类添加过滤器是不够的:您还必须定义它。这归结为添加

<filter-def name="Foo_Nondeleted"></filter-def>

到 .hbm.xml 文件。请注意这里有一个问题:虽然教程在类之前显示它,但它必须放在 XML 中的它们之后,否则您将获得一个XmlSchemaValidationException.

还有一个小改动也需要做:即使您可能已经query.substitutions设置为 map falseto 0,但它并没有应用于过滤条件,因此您必须将过滤器更改为

<filter name="Foo_Nondeleted" condition="Deleted = 0" />
于 2011-09-28T10:58:44.200 回答