This is the 3rd part of my 'Business Objects' serie. In this article, I'll discuss the LazyLoadCollection. (Part 1 and 2 can be found
here and
here.
Introduction
Suppose you have a class
Customer. This
Customer class has collection of 'Orders'. Each
Order contains a collection of
OrderLines.
Now, if you retrieve one or more Customers out of the database, it is maybe not such a good idea that you retrieve all the Orders and OrderLines with it. It is probable that the user is not interested in all the orders or all the orderlines of each customer that you have retrieved.
If you do this, it means that a lot of objects will be loaded into memory; objects you might not need.
Therefore, it is sometimes a good idea to load this information only when you need it. In the scenario I've described here, I think it is a good idea that, when you load a Customer out of the database you should also get his Orders.
In most cases, you want to have an overview of the Orders of a Customer. This means that you will need the Orders of the Customer in most of the cases.
However, you're only interested in the
OrderLines of an
Order, once you decide to view the details of an
Order. Here, you should only retrieve the OrderLines out of the database, when they're needed. This is where the Lazy Loading comes in.
The LazyBusinessObjectCollection class
The Lazy Load functionality must be transparant to the user of our
Order class. This means that, the user must be able to Add, Remove and see the OrderLines of an Order just like he/she can do with a normal collection class.
Even if the OrderLines of an Order have not been loaded yet, it must be possible that OrderLines are being added.
So, how can we do that ? The
Proxy Design Pattern offers us a way to do that.
Instead of giving the
Order class a 'real collection' as a member to hold it's orderlines, we will give it a proxy object that acts as a collection.
This means that we must create a class that has all functionality of a collection. The easy way to do so, is to create a class that is a wrapper around a Collection object, and acts as a Collection as well.
We can create a class that is a wrapper around the
BusinessObjectCollection class that we've created earlier. This wrapper class must also implement the
ICollection interface so that it can act as a collection. We also want Undo-support, so this class must implement our
IUndoable interface as well.
The skeleton of our
LazyBusinessObjectCollection will look like this:
[Serializable]
public class LazyBusinessObjectCollection<T> : IUndoable, ICollection<T>
where T : BusinessObject
{
}
Earlier, I said that the
LazyBusinessObjectCollection class would be a wrapper around the
BusinessObjectCollection class. This means that the
LazyBusinessObjectCollection class should have a member of type
BusinessObjectCollection.
[Serializable]
public class LazyBusinessObjectCollection<T> : IUndoable, ICollection<T>
where T : BusinessObject
{
private BusinessObjectCollection _collection = null;
private BusinessObjectCollection CollectionObj
{
if( _collection == null )
{
// initialize and load the items.
}
return _collection;
}
}
I've also created a property that accesses the
BusinessObjectCollection member. This property should be used everytime we access our
BusinessObjectCollection from within the
LazyBusinessObjectCollection class, since this property will take care that the
BusinessObjectCollection is loaded if this is necessary.
To be able to populate the collection, we need something that can loads the data for us. We want to decouple the
LazyBusinessObjectCollection from the code that retrieves the items for us, since, this 'data retrieval code' will be most likely in some kind of Data Access Layer.
So, I decided to create an interface
ILazyLoaderThis interface looks like this:
public interface ILazyLoader<T> where T : BusinessObject
{
List<T> GetObjects();
}
This interface defines only one method that gives us a list of BusinessObjects. This means that the class that implements this interface is responsible for retrieving and returning the BusinessObjects.
So, we can give an
ILazyLoader to our
LazyBusinessCollection class. This
ILazyLoader can be provided by the constructor of the
LazyBusinessObjectsCollection class, and the property that we've created earlier, can access the
GetObjects method which will give the Objects that the collection must contain:
[Serializable]
public class LazyBusinessObjectCollection<T> : IUndoable, ICollection<T>
where T : BusinessObject
{
private BusinessObjectCollection _collection = null;
private ILazyLoader<T> _loader;
private BusinessObjectCollection CollectionObj
{
if( _collection == null )
{
_collection = new BusinessObjectCollection<T> ();
foreach( T item in _loader.GetObjects () )
{
_collection.Add (item);
}
}
return _collection;
}
public BusinessObjectCollection( ILazyLoader<T> loader )
{
if( loader == null )
{
throw new ArgumentException ("You must provide an ILazyLoader",
"loader");
}
_loader = loader;
}
}
Pretty simple huh ?
Now, an ILazyLoader can look like this:
public OrderLinesLoader : ILazyLoader<OrderLine>
{
private int _orderId;
public OrderLinesLoader( int orderId )
{
_orderId = orderId;
}
public List<OrderLine> GetObjects()
{
// Just for simplicity of the example, normally, you should
// use parametrized queries, etc.... Consider this as pseudo-code
string sql = "SELECT * FROM orderlines " +
"WHERE OrderId = " + _orderId.ToString();
DataReader dr = dbHelper.Execute (sql);
List<OrderLine> lines = new List<OrderLine>();
while( dr.Read() )
{
OrderLine ol = CreateOrderLineFromReader (dr);
lines.Add (ol);
}
return lines;
}
}
So, the repository or factory who gives us an
Order object, is responsible for initializing the
OrderLines member -which is an
LazyBusinessObjectsCollection of the
Order class.
We still have to implement the
ICollection and
IUndoable interface. I will not put the implementation of the
ICollection interface here to save some space; besides, it is pretty simple code: you can just delegate these calls to the
BusinessObjectCollection property (note: make sure to use the property) of the
LazyBusinessObjectCollection class.
The same applies to the implementation of
IUndoable: just delegate the calls to the
BusinessObjectCollection. However, here we can use the field instead of the property, since we only have to delegate these calls if the collection has been loaded:
public void CreateSnapshot()
{
if( _collection != null )
{
_collection.CreateSnapshot ();
}
}
...