NHibernate异常:方法添加应该是'public / protected virtual'或'protected internal virtual'

时间:2010-02-17 03:27:31

标签: nhibernate lazy-loading

以此课为例:

public class Category : PersistentObject<int>
{
    public virtual string Title { get; set; }
    public virtual string Alias { get; set; }

    public virtual Category ParentCategory { get; set; }
    public virtual ISet<Category> ChildCategories { get; set; }


    public /*virtual*/ void Add(Category child)
    {
        if (child != null)
        {
            child.ParentCategory = this;
            ChildCategories.Add(child);
        }
    }
}

当运行没有add方法的虚拟关键字的应用程序时,我收到此错误:

method Add should be 'public/protected virtual' or 'protected internal virtual'

我理解为什么属性需要声明为虚拟属性,因为它需要被延迟加载功能覆盖。

但是我不明白为什么需要将方法声明为虚拟...它们需要被覆盖的原因是什么?

这非常令人困惑!

2 个答案:

答案 0 :(得分:10)

方法也需要是虚拟的,因为他们可以访问字段。考虑一下这种情况:

class Entity
{

  //...
  private int a;
  public virtual int A 
  { 
     get { return a; }
  }

  public virtual void Foo()
  {
    // lazy loading is implemented here by the proxy
    // to make the value of a available
    if (a > 7) 
    // ...
  }
}

答案 1 :(得分:4)

我认为这是NHibernate中延迟加载功能所必需的,NHibernate创建实体的代理并控制对它的所有访问。这就是每个方法和属性必须是虚拟的原因。基本上,如果有成员与实体做任何事情,NH需要了解它并利用它。 如前所述,为了让NHibernate执行“魔术”,它会创建从您的实体继承的代理类(在您的情况下为Category)。但是,如果使实体实现接口,它将使用该接口创建代理而不是具体类型。这样,您就不必将所有内容标记为虚拟。

编辑:一些更正...根据this,我不得不说,看起来NH似乎并没有真正对虚拟方法做任何事情,毕竟。我甚至读过有人说他们从NH核心组件中删除了这个运行时检查,只是为了绕过它。我的假设是,这是一个尚未删除的旧要求。很酷的是,看起来有一个主动将PostSharp用于静态代理,所以你的类不必为NH生成任何虚拟的东西来生成代理。不好的是,它看起来已经被困在分支机构中近两年了。