Click here to Skip to main content
Licence 
First Posted 16 Apr 2007
Views 72,673
Downloads 1,542
Bookmarked 104 times

Simple NHibernate Architecture

By | 16 Apr 2007 | Article
An article showing a nice architecture I've come up for using NHibernate

Introduction

I started using NHibernate at the end of last year and I am having a wonderful experience with it. The simple fact that I don't have to maintain hundreds of procedures and data access abstraction classes to do basic CRUD operations is enough to justify the use of NHibernate (even though I used a custom code generation tool). Besides that, my code is amazingly clearer and simpler. I only need to worry about business and user interface logic now. All that pluming code has gone for good.

I had zero experience with ORM frameworks, so on the first couple of weeks using NHibernate I had a hard time with some issues, like using it with ASP.NET disconnected objects and abstracting NHibernate sessions from the domain layer. After putting in some effort I came out with a nice architecture that abstracts the NHibernate layer and allows me to work with an ASP.NET session.

Background

I am assuming you already have some familiarity with NHibernate. If you are new to NHibernate I encourage you to go to the website and download the reference documentation. I am using NHibernate 1.2 CR1 in this article.

The Architecture

The idea here is to keep my business objects unaware of an NHibernate session. So I created a different project for NHibernate stuff, called Shared.NHibernateDAL. It includes the session and transaction manager classes. I also created a common business object that serves as a base class for all my business entities. This class (BusinessObject) encapsulates the common methods and properties all entities that need to be persisted have, like Save(), Delete(), ID, etc.

Here's a class diagram just to give an idea:

Screenshot - nh_classdiagram.jpg

The Implementation

The main classes on the NHibernateDAL project are the following:

  • NHibernateSessionManager – This is the class responsible for managing the NHibernate session. I've got this class from another NHibernate article on Code Project.

  • TransactionBlock – This is a class I created to manage transactions and abstract it from NHibernate. I will show an example of its use further on this article.

  • BusinessObject<T> - This is the base class for all the business entities I need to persist. It implements the basic CRUD methods.

Here's the code:

namespace Shared.NHibernateDAL
{
    public class BusinessObject<T>
    {
        public virtual void Save()
        {
            ISession session = NHibernateSessionManager.Instance.GetSession();
            if (!this.IsPersisted)
                session.Save(this);
            else
                session.SaveOrUpdateCopy(this);

            session.Flush();
        }

        public virtual void Delete()
        {            
            ISession session = NHibernateSessionManager.Instance.GetSession();
            session.Delete(this);
            session.Flush();
        }

        public virtual T Clone()
        {            
            return this.Clone();
        }

        /// <summary>
        /// Returns a valid populated object or a null reference        
        /// </summary>
        public static T Get(int id)
        {
            if (id <= 0)
                return default(T);
            else
            {
                ISession session = 
                    NHibernateSessionManager.Instance.GetSession();
                return session.Get<T>(id);
            }
        }

        private int _id;

        public virtual int ID
        {
            get { return _id; }
            set { _id = value; }
        }

        public virtual bool IsPersisted
        {
            get
            {
                return this.ID > 0;
            }
        }
    }
}

Using the Code

On this sample we are going to use a Product class to show the features.

Here's the Product class code:

namespace Business
{
public class Product : BusinessObject<Product>
{
    public Product()
    {
        this._name = null;
        this._weight = 0;
    }

    private string _name;
    private decimal _weight;

    public virtual decimal Weight
    {
        get { return _weight; }
        set { _weight = value; }
    }

    public virtual string Name
    {
        get { return _name; }
        set { _name = value; }
    }

    public static List<Product> Select()
    {
        ISession session = NHibernateSessionManager.Instance.GetSession();
        IQuery query = session.CreateQuery("from Product");
        return (List<Product>)query.List<Product>();
    }
}
}

Because all CRUD functionality is encapsulated on the BusinessObject, the domain entities are as simple as that. Now our Product class is ready to be consumed.

Note that I have included a static Select method that returns a list of products. This might be useful for populating user interface controls, but if you want to completely abstract NHibernate from your domain objects you should create another layer of abstraction under the business layer, containing all HQL queries. But, for small projects, I think it is OK to use HQL inside business objects.

The download file contains three projects, corresponding to the three layers I am using: NHibernateDAL, Business and Web. The Web project is a simple web page showing how it's easy to consume CRUD operations from the Product class.

Transactions

It is often necessary to use nested transactions with objects. You might have a method that opens its own transaction and sometimes needs to be called from another transaction. To solve this problem and also abstract the NHibernate transaction methods I've come out with the TransactionBlock class. This class keeps track of how many transaction blocks were open and ensures that only one transaction is opened on the database. At the end it commits the transaction only if all blocks were declared valid. Here's an example of its usage:

public void OutterTransaction()
{
    using (TransactionBlock tran = new TransactionBlock())
    {
        Product p1 = new Product();
        p1.Name = "Product 1";
        p1.Weight = 10;
        p1.Save();

        //Call another method with a transaction block
        InnerTransaction();

        tran.IsValid = true;
        // On the dispose call the database transaction is commited 
        //if all transaction blocks are valid
    }
}

private void InnerTransaction()
{
    using (TransactionBlock tran = new TransactionBlock())
    {
        Product p2 = new Product();
        p2.Name = "Product 2";
        p2.Weight = 10;
        p2.Save();                
        tran.IsValid = true;
    }
}

Conclusion

NHibernate has definitely helped me improve the quality and readability of my code. Combined with code generation tools (like MyGeneration), it can really improve productivity. But be aware. NHibernate and the architecture I am using will not fit every project. Working with legacy database can be a pain, although version 1.2 supports communication with stored procedures.

Any feedback will be greatly appreciated.

License

This article has no explicit license attached to it but may contain usage terms in the article text or the download files themselves. If in doubt please contact the author via the discussion board below.

A list of licenses authors might use can be found here

About the Author

Cassio Mosqueira

Software Developer (Senior)

Canada Canada

Member

I've been developing .NET enterprise applications since 2000.
 
I am originally from Rio de Janeiro and I am currently working at The Intelligent Coder Inc. in Ontario.

Sign Up to vote   Poor Excellent
Add a reason or comment to your vote: x
Votes of 3 or less require a comment

Comments and Discussions

 
You must Sign In to use this message board. (secure sign-in)
 
Search this forum  
 FAQ
    Noise  Layout  Per page   
  Refresh
GeneralMy vote of 4 PinmemberShahriar Iqbal Chowdhury22:54 2 Jul '10  
QuestionI test the code but I have a problem Pinmemberbenkhaled lotfi5:38 26 Mar '09  
GeneralNew to NHibernate where is the NHibernate database file Pinmemberarigney17:51 12 Feb '08  
GeneralWatch out for CallContext in TransactionBlock Pinmemberckinnan12:00 16 Aug '07  
GeneralRe: Watch out for CallContext in TransactionBlock PinmemberCassio Alves4:59 17 Aug '07  
GeneralRe: Watch out for CallContext in TransactionBlock PinmemberChuck Kinnan6:14 17 Aug '07  
GeneralWinforms PinmemberJuan Carlos Delgado14:13 10 May '07  
GeneralRe: Winforms PinmemberCassio Alves2:11 15 May '07  
GeneralDB Object != Business Object Pinmemberblorq9:30 24 Apr '07  
GeneralGood article, but I disagree PinmemberSean Chambers2:02 17 Apr '07  
GeneralRe: Good article, but I disagree PinmemberCraig G. Wilson2:37 17 Apr '07  
GeneralRe: Good article, but I disagree PinmemberSean Chambers3:12 17 Apr '07  
GeneralRe: Good article, but I disagree PinmemberBilly McCafferty10:29 17 Apr '07  
GeneralRe: Good article, but I disagree PinmemberCassio Alves4:55 18 Apr '07  
GeneralRe: Good article, but I disagree Pinmemberbalazs_hideghety20:30 17 Jun '07  
GeneralRe: Good article, but I disagree Pinmembernmirandaghn18:42 26 Aug '08  
GeneralRe: Good article, but I disagree Pinmembertimtas7:42 26 Apr '07  
I agree with Cassio that it depends on the project. I have used both approaches, (1) business Domain seperated from persistence layer which depends on NHibernate, and (2) ActiveRecord. I can say that ActiveRecord saves a ton of code, development time, testing etc. Of course, these savings could all be lost (and more) if you ever needed chuck NHibnerate or add some other persistence mechanism. So you need to ask yourself, based on business realities, how much am I (or my client) willing to invest now to ensure flexibility in the future? What's the likihood that this flexibility will ever be needed? And when asking this question keep in mind that a dollar today is worth a lot more than a dollar two or three years from now that might never even be spent. That said, many many cases will still indicate a more layered approach from the get go.
GeneralQuestion about B.O. to table mapping PinprotectorMarc Clifton1:13 17 Apr '07  
AnswerRe: Question about B.O. to table mapping PinmemberDavide Icardi1:47 9 May '07  

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.

Permalink | Advertise | Privacy | Mobile
Web02 | 2.5.120529.1 | Last Updated 16 Apr 2007
Article Copyright 2007 by Cassio Mosqueira
Everything else Copyright © CodeProject, 1999-2012
Terms of Use
Layout: fixed | fluid