Null Object Design Pattern

Introduction

In almost every method we write, irrespective of the programming language, we always have checks like the following one.
  1. if (objectVariable == null)  
  2.     return;  
  3. // do some work with objectVariable  
We end up adding so many null checks in our methods that it gets quite hard to figure out what we are supposed to do in our method. This makes our code look ugly and hard to read.

Scenario
 
Consider a simple console application which uses an IMobile type to perform some operations. The IMobile type is returned from a MobileRepository using a method named GetMobileByName (mobileName), where mobileName is the name of the mobile that we wish to operate on. A code implementing the above requirement would look something like the following.
  1. class Program  
  2. {  
  3.     static void Main(string[] args)  
  4.     {  
  5.         var mobileRepository = new MobileRepository();  
  6.         IMobile mobile = mobileRepository.GetMobileByName("sony");  
  7.         mobile.TurnOn();  
  8.         mobile.TurnOff();  
  9.      }  
  10. }  
Now, in this code, as we are not having a null check for the object returned by MobileRepository, we might get a NullreferenceException for some invalid input to the method GetMobileByName. In this case, we can solve this problem by simply introducing a null check for mobile.
 
But is it really a good solution? What if the GetMobileByName is being used in several methods? Would it be feasible to add a null check in every method? Well, a better alternative is to implement Null Object pattern.

What is Null Object pattern?
 
A null object is also known as a Stub, an Active Nothing or an Active Null. It helps our program logic to get rid of null checks where ever possible. We all know, that we can not call methods on a null reference as it results in a NullReferenceException. The null object pattern provides a non-functional object in place of a null reference and therefore allows methods to be called on it.
 
So, for some invalid input to the method GetMobileByName, our MobileRepository would return an instantiated, yet null, IMobile object in place of a null reference. Unlike any other Mobile object, when we perform any operation on this object, nothing would happen.

Implementation
 
As discussed above, in our scenario, we have to implement the method GetMobileByName in a way that it shall return a null IMobile object in place of a null reference, for any invalid input. In order to achieve that, we first need to create a class implementing from the IMobile interface. The object of this class would be our null object implementation. Please refer to the following code for the same.
  1. public interface IMobile  
  2. {  
  3.     void TurnOn();  
  4.     void TurnOff();  
  5. }  
  6.   
  7. //mobile type implementing IMobile interface  
  8. public class SamsungGalaxy : IMobile  
  9. {  
  10.     public void TurnOff()  
  11.     {  
  12.         Console.WriteLine("\nSamsung Galaxy Turned OFF!");  
  13.     }  
  14.   
  15.     public void TurnOn()  
  16.     {  
  17.         Console.WriteLine("\nSamsung Galaxy Turned ON!");  
  18.     }  
  19. }  
  20.   
  21. //our null object class implementing IMobile interface as a singleton  
  22. public class NullMobile : IMobile  
  23. {  
  24.     private static NullMobile _instance;  
  25.     private NullMobile()  
  26.     { }  
  27.   
  28.     public static NullMobile Instance  
  29.     {  
  30.         get {  
  31.             if (_instance == null)  
  32.                 return new NullMobile();  
  33.             return _instance;  
  34.         }  
  35.     }  
  36.   
  37.     //do nothing methods  
  38.     public void TurnOff()  
  39.     { }  
  40.   
  41.     public void TurnOn()  
  42.     { }  
  43. }  
Our repository can return different types of mobiles like SamsungGalaxy, AppleIPhone, and SonyXperia. All the types implement the IMobile interface. Our null object class, NullMobile is a classic Singleton and implements from IMobile interface. In the above code, we can see that, the NullMobile class like other types, implements the methods TurnOn and TurnOff. However, the method implementations do nothing.
 
Now, that we have our null object ready, it's time to use it at the right place. We do not want to have any null check for the type returned from the repository in the main logic. Therefore, our MobileRepository shall return a NullMobile instance in place of a null reference for any invalid input. Let's have a look how it works in code.
  1. public class MobileRepository  
  2. {  
  3.     public IMobile GetMobileByName(string mobileName)  
  4.     {  
  5.         IMobile mobile = NullMobile.Instance;  
  6.         switch (mobileName)  
  7.         {  
  8.             case "sony":  
  9.                 mobile = new SonyXperia();  
  10.                 break;  
  11.   
  12.             case "apple":  
  13.                 mobile = new AppleIPhone();  
  14.                 break;  
  15.   
  16.             case "samsung":  
  17.                 mobile = new SamsungGalaxy();  
  18.                 break;  
  19.         }  
  20.         return mobile;  
  21.     }  
  22. }  
We have first created a variable of type IMobile and initialized it with an instance of NullMobile. If the mobileName does not match a case in the switch statement, the NullMobile object is returned by the GetMobileByName method. In our main logic, when we try to call methods on this object we will not get any exception and hence the work flow will not break. However, the methods will do nothing.

Summary
  • The null object pattern helps us to write a clean code avoiding null checks where ever possible.
  • Using the null object pattern, the callers do not have to care whether they have a null object or a real object.
  • It is not possible to implement null object pattern in every scenario. Sometimes, it is likely to return a null reference and perform some null checks.
  • In above code sample, we are using an null object implementing an interface. However, we can have a quite similar approach with Abstract base class also.
  • Null object is often a singleton. However, sometimes we want our object state to vary over instances.
  • Null pattern is helpful in situations where we want to return an object of the expected type, yet do nothing.
  • It is important to note that unless developers are aware that the null object implementation exists, they may still do null checks.
I hope this helps you get a basic understanding of the null object pattern and how it helps you write a clean code. It’s always great to have feedback from the readers. Your valuable feedback, questions, or comments about this article are always welcome.


Similar Articles