Greetings! In this first article of the Timeless .NET Engineer Newsletter, we will look into an evergreen design pattern: Decorator. It’s one of the oldest patterns, as it was already mentioned in the famous Gang of Four book. It allows you to extend the functionalities of a component without altering its code. Today, we will the primary techniques for implementing the decorator pattern in modern .NET while adhering to the Single Responsibility Principle (SRP) and avoiding boilerplate code. If this email does not look good, open it in your browser. When to use a Decorator pattern in C#The Decorator pattern is useful when you want to add behavior to an existing component but either cannot or do not want to modify the source code. This is typically done to adhere to the single responsibility principle (SRP) to keep our code clean, readable, and maintainable. Some real-world use cases for the decorator design pattern include:
What is the Decorator patternA Decorator is essentially a wrapper that implements the same contract as the entity it’s wrapping. We are intentionally using the vague term contract. As we will see in this article, it can mean two things: a C# interface if we implement a type decorator, or a method signature if we implement a method decorator. In both cases, the caller does not need to know that it is talking to a decorator rather than the final implementation. The pattern is recursive: we can add a decorator to a decorator, creating a chain of responsibility. For instance, instead of just calling an unreliable service, we may want to retry a couple of times upon transient failure, and finally assign a unique ID to each exception, log it, and wrap the exception. We can represent the chain as follows: In this article, we will explore two kinds of decorators: type decorators and method decorators. The classic Type Decorator patternThe classic type decorator pattern is a purely object-oriented variant of the decorator pattern that relies on type interfaces. To illustrate the idea, let’s say we want to build a simple messaging app. We’d need a component that handles sending and receiving messages. This component implements the
We are using the
We are instantiating the
That all works nicely on our development environment. However, as soon as we move things to production, we realize that the messenger service is unreliable and occasionally causes our app to crash. Since we don’t own the source code of the How does the Decorator pattern help us tackle this problem? Take a look at our design using the type decorator pattern in the class diagram below. Along with the decorators for error handling and retrying, we’ve introduced the Here is the implementation of the
The Now, instead of passing the original
When the program calls Using type decorators with dependency injectionObviously, in any modern C# application, you would not instantiate the components manually as in the examples above, but you would let dependency injection do the job. If you are using .NET Core’s IServiceCollection, there is a nice library called Scrutor that can easily wrap a service with a decorator. For example, this is how to apply the decorators by using Scrutor. Note the calls to the
Many dependency injection frameworks have built-in support for decorators. See for instance how Autofac handles this problem, The Abstract Type Decorator patternAt first glance, our solution design seems perfect. But when we dig into the implementation of the decorators, we notice that the exception handling would be duplicated across other methods. Here, we’re violating the Don’t Repeat Yourself principle. The code is now harder to maintain than before because any change to the error handling must be made in each method of We will now see how to improve the Type Decorator pattern to make the decorator logic more reusable. Let’s use the word policy to designate the logic that we wrap a method call with. Policies can be abstracted away and encapsulated in a reusable way. In the following diagram, we have represented policies as an interface. Here is the exception-reporting policy:
We then define an
In practice, you’ll also need to implement With this setup, all the
Note that this decorator is now abstracted from any policy. The only repetitive code is now in calling the Finally, we wire the service collection using Scrutor’s
We’ve now consolidated error-handling logic in one place. The control flow now becomes: Generating Type Decorators automaticallyThere’s still repetitive code in the
In this article, we will only explore the first solution. The principle behind dynamic proxies is to generate the decorator class at run time, when the application initializes. Among the few libraries that implement this feature, the most popular is Castle DynamicProxy. The concept of policy developed earlier maps to Castle’s
As promised, there’s no longer a need for any decorator code since Castle has implemented it. We can now proceed to the startup sequence of our application. We will need a
We can now use the
The Method Decorator PatternSo far, we’ve discussed techniques that help replace a type with another type implementing the same interface, but providing additional services. The principal benefits of this approach are:
However, there’s a significant drawback: it only works if you can inject yourself into the communication between the caller and the service – typically through an interface, although the same could be achieved using An alternative to the type decorator pattern is the method decorator. As its name suggests, method decorators target a single method and not the entire type. Method decorators are commonly used in dynamic languages such as Python. They aren’t directly supported by C#, but some toolkits like Metalama make it possible. The idea of C# method decorators is to move the logic of the policy to a special kind of custom attribute called an aspect, which could be compared to code templates. Unlike other custom attributes, aspects are applied to the code during compilation. Since this approach is compile-time, we’re not limited to the limitations of the .NET runtime, namely we’re not limited to virtual or interface methods, but we can intercept anything (including static private fields, if you ask). Here is the Metalama version of the retry policy:
To add the policy to a method, apply it as a custom attribute:
To further improve maintainability, toolkits like Metalama facilitate the bulk application of aspects, eliminating the need for developers to manually specify where each aspect should be used. For instance, we can stipulate that all public methods within a specific namespace should have exception reporting. Consequently, when new methods are added to this namespace, the exception-reporting aspect is automatically applied. This approach not only enhances the readability and maintainability of the codebase but also simplifies scalability. In Metalama, this is achieved using fabrics. The following example demonstrates how to add exception reporting to all public methods in a project:
SummaryDecorators are an effective way to maintain clean code and uphold the single responsibility principle. Opt for type decorators when you don’t own the code that you wish to enhance with new behaviors, or when you need to dynamically add behaviors at runtime. Use method decorators (aspect-oriented) when you own the source code and aim to adhere to the Single Responsibility Principle. This article was simultaneously published on the PostSharp Blog under the title: The Decorator Pattern in Modern C#. |
Learn the art of building robust and maintainable applications with less code. This newsletter does not focus on shiny C# features or the latest .NET APIs, but instead on the timeless art of building stuff that lasts, code that you will be proud of even 10 years from now.
Fast and Compact Structured Logging in C# Using String Interpolation Greetings! In this iteration of the Timeless .NET Engineer newsletter, we are looking into some practical approaches to structured logging. You probably already know structured logging and C# string interpolation (and I wonder how we could live without that). But did you know that string interpolation and logging don’t play well together by default, both because it “cancels” the benefits of structured logging, and because it...