How you can create extension methods in C#
Extension methods in C# are really useful, but there are few rules to follow…
Table of Contents
Just a second! π«·
If you are here, it means that you are a software developer. So, you know that storage, networking, and domain management have a cost .
If you want to support this blog, please ensure that you have disabled the adblocker for this site. I configured Google AdSense to show as few ADS as possible - I don't want to bother you with lots of ads, but I still need to add some to pay for the resources for my site.
Thank you for your understanding.
- Davide
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}";
}
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());
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
}
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}";
}
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.
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);
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 behavior 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()}";
}
You can extend subclasses
Say that you have a Student
class that extends Person
:
public class Student : Person
{
public int StudentId { get; set; }
}
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}";
}
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]}.";
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());
will print out
1971-06-28 - Musk Elon
Musk Elon - ID: 123
Student: M.E.
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();
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
}
}
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()));
}
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());
This article first appeared on Code4IT
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.