This is intended to be a collection of most of the well-known design patterns, with explanation, examples, illustrations, and everything you need to learn and understand design patterns without the trivial examples usually found while explaining the patterns.
Design patterns are gaining more popularity each day, but the problem is that they are often misused, either by putting them outside of their intended context, or by putting them in the right position, with the correct usage, but before the problem is frequent or big enough to deserve implementing a known design pattern which will often end by you over-engineering the code base and increasing its complexity.
One of the causes that will result in misusing the patterns is that most of the examples found online to learn patterns are often very simple and trivial, that you won't be able to understand when to use the pattern or why it existed in the first place.
In this collection of patterns I'll try as possible to stay away of any ducks and pizzas examples, and write real cases while explaining when to use the pattern, and what problems implementing the pattern will introduce.
There's no correct way to consume this content, you can go through the patterns with any order, and if a pattern uses a concept from another pattern, or the example make use of another pattern, I'll make sure to mention it.
I can recommend to read and understand the SOLID Principles before moving to any of the patterns.
You can choose any of the patterns from the patterns catalogue.
- Design patterns can be easily misused, so make sure to fully understand the pattern, its use cases, and what is the problem it is trying to solve.
- Design patterns don't have one way for implementation, you can change the implementation, the components, and even mix between the patterns as you wish.
- Design patterns aren't the ultimate solution to a problem, they may even introduce a few problems of their own due to their limitations, so implement only what you need, and feel free to extend them with what you see fits your context.
- I'll try as possible to avoid trivial examples, but I may not be able to make real-life examples for all the patterns, so feel free to contribute with a better example, or try searching one of the learning sources for better examples.
- Please be careful as this code is not production quality by any means, avoid copying and pasting or using the code in the examples as is, as it is not intended to be used in production code.
As you can see, all the examples and tests are written in TypeScript. There's no particular reason for me to use TypeScript while explaining Design Patterns other than I'm familiar with it.
Some of the patterns heavily depend on certain language features which may not be found in TypeScript, or can easily be mis-implemented without a certain feature, it may behave the same, but sometimes it will lack the complete features or benefits the same pattern will provide in other languages.
I will try as possible to be clear on where TypeScript falls short, but maybe in the future I'll implement the examples in other languages.
Finally, feel free to contribute to the code with other examples or even examples written in other languages.
You'll need this only if you want to play with the code and try new things, or if you want to contribute to the project.
# clone the repository
$ git clone https://github.com/khaled-hamam/design-patterns-explained.git
# Installing the needed dependencies
$ npm install
# OPTIONAL: Run Compilation for Error Checking
$ npm run build
# OPTIONAL: Run Compilation on Watch Mode
$ npm run build:watch
# Running the tests
$ npm test
# Running the tests in watch mode
$ npm run test:watch
# Running tests for a certain pattern
$ npm run test <pattern-folder-name>
# Running Linting on Markdown Files
$ npm run lint
# Auto Fix basic errors in Markdown
$ npm run lint -- --fix
# Format files with prettier
$ npm run format