tags:

views:

56

answers:

3

I have read in Chapter 4 of the NHibernate docs that all of a persistent classes public methods, properties and events must be declared as virtual.

However, whilst a runtime error is generated for any Properties that are not marked as virtual, I have found that static methods are allowed and do not generate a runtime error . As they are static they are of course not marked virtual which seems to break the rule in point 4.1.4 of the documentation (see above). I have checked the resulting sql and it also implements lazy loading correctly when I run a test against the method so is it therefore ok to use static methods?

Here's the basic details of the persistant class:

public class CmsPage
{
    public virtual int? Id { get; set; }
    public virtual string Title { get; set; }

    public virtual void Update()
    {
        using (ISession session = NHibernateHelper.OpenSession())
        {
            using (ITransaction transaction = session.BeginTransaction())
            {
                session.Update(this);
                transaction.Commit();
            }
        }
    }

    // Note: static and non-virtual and yet it will not cause a problem for Nhibernate
    public static IEnumerable<CmsPage> GetList()
    {
        IList<CmsPage> pageList;
        using (ISession session = NHibernateHelper.OpenSession())
        {
            string hql = "from CmsPage p";
            pageList = session.CreateQuery(hql)
                .List<CmsPage>();
        }

        return pageList;
    }
}

So my question is why is it ok to use a static method in the persistent domain class when the documentation seems to say it's not?

Please answer from NHibernate's point of view not an OO design point of view; I don't want to get into an OOD/OOP debate if it can be avoided please.

A: 

Actually this applies only to properties. Methods are not persisted, so proxies and lazy-loading does not apply. Ideally you should separate data access (the static methods in your case) from the domain object. But you are correct to point this out, maybe the documentation should have been clearer.

In conclusion your class is perfectly fine but it could be even better if you separated the concerns.

Darin Dimitrov
hmmm did wonder if it might just be properties regardless of the documentation. Thanks.I've kept them in the domain object as it's not really data access in my opinion. Traditionally I would have always had a BLL DAL etc but it's NHibernate that is doing the data access really. Although HQL might not be that welcome in the Domain layer I admit... not worked through that in my head yet. I toyed with having it all in Repository classes but it left me with an anaemic domain model. I think I'll probably end up with stubs in the Domain that call stuff in the repo classes in the end.
Mr Grok
+1  A: 

The documentation says: "NHibernate works best if these classes follow some simple rules, ..." It doesn't say it won't work (clearly it does work).

So, really, the discussion boils down to an OO issue.

Jon Seigel
True! scan reading is as always my enemy...
Mr Grok
A: 

NHibernate needs all your properties to be virtual because it carries out its lazy-loading magic by making proxies of your objects that override everything. So when you write this code:

class Foo {
    public virtual Foo[] Neighbors { get; set; }
}

NHibernate secretly generates classes like:

class NHProxy03450843275 : Foo {
    public virtual Foo[] Neighbors { /* Godawful lazy-loading magic goes here */ }
}

Actually it's worse than that, but this gives you the idea. Anyway, static methods aren't bound to particular instances of a class, so NH doesn't need proxies to deal with them. Thus they can be non-virtual.

David Seiler