Template Pattern C#

đź‘‹ About the Author

Hi, I’m Shadman Kudchikar, the author of this blog post! I specialize in .NET development and have a passion for building scalable, high-performance applications.

I am also the creator of BytLabs.MicroserviceTemplate, a comprehensive template designed to help developers quickly build robust microservices with best practices.

I am currently exploring new opportunities as a Lead Developer in .NET. If you're looking for someone to bring expertise in designing robust solutions to your team, I’d love to hear from you!

Feel free to reach out to me directly on LinkedIn to discuss potential opportunities. Let’s connect and explore how I can contribute to your success!

Contents

What Is Template Method Pattern?

The Template Method pattern in C# enables algorithms to defer certain steps to subclasses. The structure of the algorithm does not change, but small well-defined parts of its operation are handled elsewhere.

Template Methods are very useful in conjunction with the Strategy pattern. Any system that wishes to defer primitive operations to other classes will benefit from this pattern. In C#, it is used extensively through predefined interfaces.

The Template Method pattern defines the skeleton of an algorithm in an operation, deferring some steps to subclasses. Template Method lets subclasses redefine certain steps of an algorithm without changing the algorithm’s structure.

Template Method Pattern C# Example

Let’s start with the problem statement,

Let say you are creating a library of algorithms for C# developers that contain all type of sorting algorithm. One of the algorithms would be Merge Sort like this:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
using System.Collections.Generic;
using System.Linq;

public class MergeSort
{
    public List<int> Sort(List<int> unsorted)
    {
        if (unsorted.Count <= 1)
            return unsorted;

        List<int> left = new List<int>();
        List<int> right = new List<int>();

        int middle = unsorted.Count / 2;
        for (int i = 0; i < middle; i++)  //Dividing the unsorted list
        {
            left.Add(unsorted[i]);
        }
        for (int i = middle; i < unsorted.Count; i++)
        {
            right.Add(unsorted[i]);
        }

        left = Sort(left);
        right = Sort(right);
        return Merge(left, right);
    }

    private List<int> Merge(List<int> left, List<int> right)
    {
        List<int> result = new List<int>();

        while (left.Count > 0 || right.Count > 0)
        {
            if (left.Count > 0 && right.Count > 0)
            {
                //Comparing First two elements to see which is smaller
                if (left.First() <= right.First())  
                {
                    result.Add(left.First());
                    //Rest of the list minus the first element
                    left.Remove(left.First());      
                }
                else
                {
                    result.Add(right.First());
                    right.Remove(right.First());
                }
            }
            else if (left.Count > 0)
            {
                result.Add(left.First());
                left.Remove(left.First());
            }
            else if (right.Count > 0)
            {
                result.Add(right.First());

                right.Remove(right.First());
            }
        }
        return result;
    }
}

Now you don’t have to know the code for the Merge Sort, it’s only going to be here to prove a point.

Let’s see how clients will use our above class,

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
using System;
using System.Collections.Generic;
using System.Linq;

namespace example1
{
    class Program
    {
        static void Main(string[] args)
        {
            Random _rand = new Random(1024);

            //Generate list of random int variables
            var unsorted = Enumerable.Range(0, 10)
                                    .Select(r => _rand.Next(100))
                                    .ToList();
            List<int> sorted;


            Console.WriteLine("Original array elements:");
            foreach (var item in unsorted)
            {
                Console.Write(item + " ");
            }
            Console.WriteLine();

            var algorithm = new MergeSort();
            sorted = algorithm.Sort(unsorted);

            Console.WriteLine("Sorted array elements: ");
            foreach (var item in sorted)
            {
                Console.Write(item + " ");
            }
            Console.WriteLine();
            Console.ReadLine();
        }        
    }
}

Everything looks cool in the above program right! If we run the above program we will get the following output.

1
2
3
4
Original array elements:
68 27 53 12 52 43 71 16 66 87
Sorted array elements:
12 16 27 43 52 53 66 68 71 87

Now,

Here’s my problem, it only stores int.

What if our clients want to use a different data type like float or string or any user-defined type like below one, a Person type.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
public class Person
{
    public Person(string name, int age)
    {
        Name = name;
        Age = age;
    }

    public string Name { get; set; }
    public int Age { get; set; }    

    public override string ToString()
    {
        return $"Name: {Name}\tAge: {Age}";
    }
}

Also, what if I have to sort the list of person based on its age or name.

One thing we can do is make our MergeSort algorithm generic like this,

Note: You can learn more about Generic Types in my blog post Generics In C#.

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
using System;
using System.Collections.Generic;
using System.Linq;

namespace example2
{
    public class MergeSort<T>
    {
        public List<T> Sort(List<T> unsorted)
        {
            if (unsorted.Count <= 1)
                return unsorted;

            List<T> left = new List<T>();
            List<T> right = new List<T>();

            int middle = unsorted.Count / 2;
            for (int i = 0; i < middle; i++)  //Dividing the unsorted list
            {
                left.Add(unsorted[i]);
            }
            for (int i = middle; i < unsorted.Count; i++)
            {
                right.Add(unsorted[i]);
            }

            left = Sort(left);
            right = Sort(right);
            return Merge(left, right);
        }

        private List<T> Merge(List<T> left, List<T> right)
        {
            List<T> result = new List<T>();

            while (left.Count > 0 || right.Count > 0)
            {
                if (left.Count > 0 && right.Count > 0)
                {
                    //Comparing First two elements to see which is smaller
                    if (left.First() <= right.First())
                    {
                        result.Add(left.First());
                        //Rest of the list minus the first element
                        left.Remove(left.First());      
                    }
                    else
                    {
                        result.Add(right.First());
                        right.Remove(right.First());
                    }
                }
                else if (left.Count > 0)
                {
                    result.Add(left.First());
                    left.Remove(left.First());
                }
                else if (right.Count > 0)
                {
                    result.Add(right.First());

                    right.Remove(right.First());
                }
            }
            return result;
        }
    }
}

After we make this adjustment you will find that there would be a compile-time error

So, how to solve this?

As I mentioned earlier, In C#, the template method is used extensively through predefined interfaces, one such interface is IComparable.

In the above sort method, the compare operations can replace <= operator with the primitive operation CompareTo, as in:

1
if (left.First().CompareTo(right.First()) <= 0)  

CompareTo is a method that returns 1 if a>b, 0 if a==b, and -1 if a<b.

It is defined in the IComparable interface, which many predefined C# types already implement. For example, CompareTo is available for integers and strings. Because our sort methods are meant to be general, we declare that the item types must implement the interface using:

1
public class MergeSort<T> : where T : IComparable<T>

The where clause puts a constraint on the type <T> by saying it must implement IComparable. Any class can do so if it provides a CompareTo method. So, inside a Person class, we have to implement the IComparable interface like this:

1
2
3
4
5
6
7
8
9
10
11
public class Person : IComparable<Person> 
{
    ...
    ...

    public int CompareTo(Person other)
    {
        // Comparison based on names
        return Name.CompareTo(other.Name);
    }    
}

But how is CompareTo defined in terms of CompareTo? Well, we are looking at names, so the name will presumably be a string. CompareTo is defined for strings so that is what will be called. The check for the type of the object is necessary because the implementation of CompareTo must match its interface, which is defined on the object.

Let’s see how clients can use our new update merge sort class,

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
using System;
using System.Collections.Generic;
using System.Linq;

namespace example2
{
    class Program
    {
        public class Person : IComparable<Person>
        {
            public Person(string name, int age)
            {
                Name = name;
                Age = age;
            }

            public string Name { get; set; }
            public int Age { get; set; }

            public int CompareTo(Person other)
            {
                return Name.CompareTo(other.Name);
            }

            public override string ToString()
            {
                return $"Name: {Name}\tAge: {Age}";
            }
        }

        static void Main(string[] args)
        {
            var names = new List<string>()
            {
                "Tamra Grist"       ,
                "Bennie Sweatt"     ,
                "Misha Mattei"      ,
                "Mable Lampkins"    ,
                "Kaley Gervasi"     ,
                "Nettie Horace"     ,
                "Cassidy Broxton"   ,
                "January Berk"      ,
                "Michele Barga"     ,
                "Arden Emig"        ,
            };

            Random _rand = new Random(1024);

            //Generate list of random int variables
            var unsorted = Enumerable.Range(0, 10)
                                    .Select(r => new Person(names[r], _rand.Next(100)))
                                    .ToList();
            List<Person> sorted;


            Console.WriteLine("Original array elements:");
            foreach (var item in unsorted)
            {
                Console.WriteLine(item);
            }
            Console.WriteLine();

            var algorithm = new MergeSort<Person>();
            sorted = algorithm.Sort(unsorted);

            Console.WriteLine("Sorted array elements: ");
            foreach (var item in sorted)
            {
                Console.WriteLine(item);
            }
            Console.WriteLine();
            Console.ReadLine();
        }
    }
}

If we run the above program we will get the following output:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
Original array elements:
Name: Tamra Grist       Age: 68
Name: Bennie Sweatt     Age: 27
Name: Misha Mattei      Age: 53
Name: Mable Lampkins    Age: 12
Name: Kaley Gervasi     Age: 52
Name: Nettie Horace     Age: 43
Name: Cassidy Broxton   Age: 71
Name: January Berk      Age: 16
Name: Michele Barga     Age: 66
Name: Arden Emig        Age: 87

Sorted array elements:
Name: Arden Emig        Age: 87
Name: Bennie Sweatt     Age: 27
Name: Cassidy Broxton   Age: 71
Name: January Berk      Age: 16
Name: Kaley Gervasi     Age: 52
Name: Mable Lampkins    Age: 12
Name: Michele Barga     Age: 66
Name: Misha Mattei      Age: 53
Name: Nettie Horace     Age: 43
Name: Tamra Grist       Age: 68

When a Template Method uses an interface for specifying the primitive operations, the classes must implement those operations. It is also possible to define IPrimitives as an abstract class and to provide default behavior for any of the operations, leaving the subclass to override as many operations as it wishes. Methods that have defaults are called hook operations, and they often do nothing by default.

Where To Apply Template Method Pattern?

  • When you need to implement the invariant parts of an algorithm once and leave it up to subclasses to implement the behavior that can vary.

  • When common behavior among subclasses should be factored and localized in a common class to avoid code duplication. This is a good example of “refactoring to generalize” as described by Opdyke and Johnson in their research paper Creating Abstract Superclasses by Refactoring. You first identify the differences in the existing code and then separate the differences into new operations. Finally, you replace the differing code with a template method that calls one of these new operations.

  • When you need to control subclasses extensions. You can define a template method that calls “hook” operations at specific points, thereby permitting extensions only at those points.

The Template Method pattern is more like the Strategy pattern in that it is algorithm-based. The steps of the algorithm are specified in the Template Method and some are deferred to domain classes.

Note: You can download the complete solution demo from my github repository.

Further Reading

  • Strategy Pattern C# - The Strategy pattern in C# lets the algorithm vary independently from clients that use it. The Strategy pattern enables a client to choose which algorithm to use from a family of algorithms and gives it a simple way to access it.

  • Template Method Pattern. Can we do better? by Leif Battermann - In this post, Leif compare the application of the Template Method pattern to an alternative approach adopted from functional programming. This alternative approach made use of higher-order functions. Even though this example is kind of simple and bold it demonstrates that the functional way may lead to cleaner and more maintainable code in some situations.

  • Curiously recurring template pattern by Guillermo Subirán - In this post, Guillermo explains the curiously recurring template pattern (CRTP) which is an idiom in C++ in which class X derives from a class template instantiation using X itself as a template argument. More generally it is known as F-bound polymorphism, and it is a form of F-bounded quantification. The idea of CRTP is similar to the template method pattern we discussed in this post.

Template Pattern C#
Share this

Subscribe to Code with Shadman