DEV Community

Cover image for How you can create extension methods in C#
Davide Bellone
Davide Bellone

Posted on • Originally published at code4it.dev

How you can create extension methods in C#

Probably you have already heard of extension methods: those are C# methods used to add new functionalities to an existing class.

This functionality is available since C# 3.0, so it's largely used and well documented.

The basics of extension method

Let's say that you have a non-static class that you cannot modify, maybe because it comes from an external library.

public class Person
{
    public DateTime BirthDate { get; set; }
    public string Name { get; set; }
    public string Surname { get; set; }
    public string GetFullName() => $"{Surname} {Name}";
}
Enter fullscreen mode Exit fullscreen mode

For the Person class we have only the GetFullName method.

So we can use the Person class this way:

var person = new Person() { 
    Name = "Davide", 
    Surname = "Bellone", 
    BirthDate = new DateTime(1990, 1, 1)
    };

Console.WriteLine(person.GetFullName());

Enter fullscreen mode Exit fullscreen mode

Nothing difficult, right?

Now we want to get the full name with the birthdate. We have 2 ways to achieve this: using a subclass - terrible idea - or creating an extension method. Let's go with the second approach!

First of all, we need a static class that contains our method:

public static class MyExtensions
{
    // Your methods here
}
Enter fullscreen mode Exit fullscreen mode

Now we can create the new method. We must remember 2 things:

  • it must be a static method
  • the first parameter must be of the same type we want to extend and must be preceded by the this keyword
public static string GetFullNameWithBirthDate(this Person person)
{
    return $"{person.BirthDate.ToString("yyy-MM-dd")} - {person.Surname} {person.Name}";
}
Enter fullscreen mode Exit fullscreen mode

Now we can use it with person.GetFullNameWithBirthDate(). Easy-peasy.

If you use Visual Studio and you have the Intellisense enabled, you will see those hints, both on the icon and on the description of the method.

Intellisense with extension method

Behind the scenes, we are calling the method on the MyExtensions class, not on the person object. Extension methods act just as shortcuts.

person.GetFullNameWithBirthDate()
MyExtensions.GetFullNameWithBirthDate(person);
Enter fullscreen mode Exit fullscreen mode

Ok, we have seen an overview of this functionality. Let's go more in details.

Visibility matters

As you may imagine, since extension methods are meant to extend the public behaviour of a class, you cannot use a private field (or method) of that class.

So, if we have a private string ID to our Person class, it won't be accessible from the extension method.

Of course, you can use both public properties and public methods. In fact, we can do refactor GetFullNameWithBirthDate to

public static string GetFullNameWithBirthDate(this Person person)
{
    return $"{person.BirthDate.ToString("yyy-MM-dd")} - {person.GetFullName()}";
}
Enter fullscreen mode Exit fullscreen mode

You can extend subclasses

Say that you have a Student class that extends Person:

public class Student : Person
{
    public int StudentId { get; set; }

}
Enter fullscreen mode Exit fullscreen mode

Now, among our extension methods, we can add functionalities to Student, and reference the Person class:

public static string GetFullStudentInfo(this Student student) {
    return $"{student.GetFullName()} - ID: {student.StudentId}";
}
Enter fullscreen mode Exit fullscreen mode

Finally, we can create extension methods that will override the ones created for the base class:

public static string GetInitials(this Student student)
    => $"Student: {student.Surname.ToCharArray()[0]}.{student.Name.ToCharArray()[0]}.";

public static string GetInitials(this Person person)
    => $"Person: {person.Surname.ToCharArray()[0]}.{person.Name.ToCharArray()[0]}.";
Enter fullscreen mode Exit fullscreen mode

So, now

var student = new Student() {
    Name = "Elon",
    Surname = "Musk",
    BirthDate = new DateTime(1971, 6, 28),
    StudentId = 123 };

Console.WriteLine(student.GetFullNameWithBirthDate());
Console.WriteLine(student.GetFullStudentInfo());

Console.WriteLine(student.GetInitials());
Enter fullscreen mode Exit fullscreen mode

will print out

1971-06-28 - Musk Elon
Musk Elon - ID: 123
Student: M.E.
Enter fullscreen mode Exit fullscreen mode

A real life example: LINQ

Since C# 3.5 we have a new query language: LINQ.

LINQ stands for Language-Integrated Query, and it is often used when you are working on collections and you need to do some operations on them, like get all the values that match a particular condition or take only the first element of the collection.

An example can be

List<int> myList = new List<int>() { 2, 4, 82, 6, 223, 5, 7, 342, 234, 1};
myList.First();
Enter fullscreen mode Exit fullscreen mode

By default, List<T> does not include a method that returns the first element of the collection. To use it, we need to include LINQ by putting using System.Linq among the imports.

So yes, we can say that First is a method that extends the List class: LINQ is a set of extension methods!

In fact, we can read its source code and notice this:

namespace System.Linq
{
    public static partial class Enumerable
    {
        public static TSource First<TSource>(this IEnumerable<TSource> source)
        {
            TSource first = source.TryGetFirst(out bool found);
            if (!found)
            {
                ThrowHelper.ThrowNoElementsException();
            }

            return first;
        }
        // other methods
    }
}
Enter fullscreen mode Exit fullscreen mode

So, you can extend this library if you want. Let's say that you need to get a random element from the list; you can create a method like this:

public static T GetRandom<T> (this IEnumerable<T> source)
{
if (source == null || source.Count() == 0)
    throw new ArgumentException(nameof(source)) ;
Random rd = new Random();
return source.ElementAt(rd.Next(0, source.Count()));
}
Enter fullscreen mode Exit fullscreen mode

And call it this way:

List<int> myList = new List<int>() { 2, 4, 82, 6, 223, 5, 7, 342, 234, 1};
Console.WriteLine(myList.GetRandom());
Enter fullscreen mode Exit fullscreen mode

Wrapping up

Have you ever used extension methods for creating a library?

Anyway, here's a short recap of the main points:

  • all in a static class
  • each method must be static
  • the extended class must be non-static
  • the first parameter must be preceded by the this keyword
  • you can use only public fields and methods
  • you can extend classes, but also subclasses and structs
  • the greatest example of library based on extension methods is LINQ

You can see the full example here.

Top comments (0)