Dry antipattern
For a long time I wondered what was wrong with some parts of the code. Time after time, in each of the projects there is a certain “particularly vulnerable” component, which “falls down” all the time. The customer has the ability to periodically change the requirements, and the canons of agile bequeathed to us all Wishlist to implement, running change requests in our scrum-mechanism. And as soon as the changes concern this component, in a couple of days QA find several new defects in it, rediscover the old ones, or even report its complete inoperability at one of the points of application. So why is one of the components on the lips all the time, why is the phrase a la “again # the component # broke” so often said? Why is this component cited as an example, in the context of “just not get another one like that”? What makes this component so unstable to change?
When they find the reason that led to, or contributed to the development of such a defect in the application, this reason is designated as antipattern.
This time, the Strategy pattern became a stumbling block. Abuse of this pattern has led to the creation of the most fragile parts of our projects. The pattern itself has a completely “peaceful” application, the problem is rather that it is put where it fits, and not where it is needed. “If you understand what I mean” (c).
The pattern exists in several “guises”. Its essence does not change much from this, the danger of its use exists in any of them.
The first, classic view is a certain long-lived object, which accepts another object via the interface, the strategy itself, through the setter, with some change in state.
The second type, a degenerate version of the first - the strategy is adopted once, for the entire life of the object. Those. for one scenario one strategy is used, for another another.
The third type is an executable method, either static or in a short-lived object, which takes an interface strategy as an input parameter. In the “gang of four” this species is called the “Template method”.
The fourth type is interface, aka UI. Sometimes referred to as a “pattern” as a pattern, sometimes as a “container”. On the example of web development, it is a certain piece of markup containing a placeholder (or even more than one) where, during execution, the variable part of the markup that has several different implementations will be rendered. Parallel to the markup, parallel view models or controllers also live in JavaScript code, depending on the architecture adopted in the application, organized in the second form.
Common features of all these cases:
1) some immutable part aka having one implementation, further, I will call it a container
2) mutable part, it’s also a strategy
3) the place of use, creating / invoking the container, determining which strategy the container should use, hereinafter I will call it a script.
At first, when a component using this pattern was only implemented in the project, it did not seem so bad. It was applied when it was necessary to create two identical pages (again, for example, web development), which only differ slightly in content in the middle. On the contrary, the developer was glad how beautiful and elegant it was to implement the DRY principle, i.e. completely avoid duplication of code. These are the epithets I heard about a component when it was just created. The one that became the popol of the whole project a few months later.
And since I started to theorize, I’ll go a little further - it is the attempts to implement the DRY principle that, through the strategy pattern, as well as through inheritance, lead to darkness. When, for the sake of DRY, without noticing it, the developer sacrifices the SRP principle, the first and main postulate from SOLID. By the way, DRY is not part of SOLID, and in a conflict, you must sacrifice it, because it does not favor the creation of a stable code, unlike, in fact, SOLID. As it turned out - rather the opposite. Reusing the code should be a pleasant bonus of certain design decisions, and not the purpose of making them.
And the temptation to reuse occurs when the customer comes with a new story to create a third page. After all, she is so similar to the first two. The desire of the customer to realize everything “cheaper” contributes a lot to this, as reusing a previously created container is faster than realizing the page completely. The story came to another developer, who quickly found out that the container’s functionality is not enough, and full refactoring does not fit into the estimates. Another mistake here is that the developer continues to follow the plan set by the estimates, and this happens “in silence”, as it were, there is no responsibility, it lies with the whole team that made such a decision and such an assessment.
And now new functionality is added to the container, new methods and fields are added to the strategy interface. Ifs appear in the container, and in old implementations of the strategy “stubs” appear so as not to break existing pages. At the time of the second story, the component was already doomed, and the further, the worse. It’s becoming more and more difficult for developers to understand how it works, including those that “just recently did something in it”. It's getting harder to make changes. Increasingly, one has to consult with "previous ones" to ask how it works, why some changes were made. It is increasingly likely that even the slightest change will introduce a new defect. Actually, we are already starting to talk about the fact that there is an increasing likelihood of introducing two or more defects, because one defect appears already with a near-unit probability. And now comes the moment when it is impossible to implement the new customer requirements. There are two ways out: either completely rewrite or make an explicit hack. And in the hangar there is just a suitable hack tool - you can emit events from the bottom up, then broadcast from top to bottom when you finished the dirty deals at the top. At the same time, technical debt is no longer increasing; it has already been equal to the cost of selling this component from scratch.
Inheritance is often criticized, and the corporation of goodness in its Go language generally decided to do without it, and it seems to me that the negative to inheritance partly comes from the experience of implementing the DRY principle through it. The "strategic" DRY also leads to sad results. Remains direct aggregation. To illustrate, I will take a simple example and show how it can be represented as a strategy, that is, a template method and without it.
Suppose we have two very similar scenarios represented by the following pseudo-code:
They repeat 10 lines of X at the beginning and 15 lines of Y at the end. In the middle, one script has lines A, and the other has lines B
Here it is assumed that all methods are in different classes.
As I said, at the time of implementation, the first option does not look so bad. Its disadvantage is not that it is initially bad, but that it is unstable to change. And yet, it is worse read, although with a simple example this may not be obvious. When you need to implement the third scenario, which is similar to the first two, but not 100%, there is a desire to reuse the code contained in the container. But it cannot be partially reused, you can only take it in its entirety, so you have to make changes to the container, which immediately carries the risk of breaking other scenarios. The same thing happens when a new requirement involves changes to scenario A, but this should not affect scenario B. In the case of aggregation, method X can easily be replaced by method X 'in one scenario, completely without affecting the others. It is easy to assume that the methods X and X 'can also almost completely coincide, and they can also be subdivided. With a “strategic” approach, if cascaded in the same “strategic” way, then the evil placed in the project is raised to the second degree.
Many examples of the use of the strategy pattern are visible and often used. They are all united by one simple rule - there is no business logic in the container. Absolutely. There may be algorithmic filling, for example a hash table, search or sorting. The strategy contains business logic. The rule according to which one element is equal to another or more or less is business logic. All linq operators are also the embodiment of a pattern, for example .Where () operator is also a template method, and the lambda it accepts is a strategy.
In addition to algorithmic filling, it can be filling connected with the outside world, asynchronous requests, for example, or, in the example of the “gang of four”, a subscription to a mouse click event. What they call callbacks is essentially the same strategy, I hope all my hyper generalizations will forgive me. Also, if it is a UI, then it can be tabs, or a pop-up window.
In a word, it can be anything completely abstracted from business logic.
If you use the strategy pattern in development, and the business logic has got into the container, you should know that you have already crossed the line, and stand ankle-deep in ... mmm, swamp.
Sometimes it’s not easy to understand where the line is between business logic and general programming tasks. And at first, when the component is just created, to determine that it will bring hemorrhoids in the future is not easy. And if business requirements never change, then this component may never come up. But if there are changes, the following code-smells will inevitably appear:
1. The number of methods passed. The parameter under discussion is not harmful in itself, but it can still hint. Two or three is still fine, but if the strategy contains a dozen methods, then something is probably wrong.
2. Flags. If in addition to methods in the strategy there are fields / properties, it is worth paying attention to what they are called. Fields such as Name, Header, ContentText are valid. But if you see fields such as SkipSomeCheck, IsSomethingAllowed, this means that the strategy is already smelling stinky.
3. Conditional calls. Associated with flags. If the container has a similar code, then you have already gone to the swamp waist-high
4. Inadequate code. Using an example from JavaScript -
From the name it can be seen that doSomething is a method, but it is checked as a flag. That is, the developers were too lazy to create a flag indicating the type, but used the presence / absence of a method as a flag, and even if it was not called inside the if block. If you encounter this, you should know that the component is already up to its debt in technical debt.
Once again, I want to express my opinion that the root cause of all that I described is not in the pattern as such, but in the fact that it was used for the DRY principle, and this principle was placed above the principle of sole responsibility, like SRP. And, by the way, I have already come across more than once that the principle of sole responsibility is somehow not quite adequately interpreted. About how "my divine class controls a satellite, to control a satellite is his only responsibility." On this note I want to finish my opus and wish less often in response to “why so”, to hear the phrase “historically so happened”.
When they find the reason that led to, or contributed to the development of such a defect in the application, this reason is designated as antipattern.
This time, the Strategy pattern became a stumbling block. Abuse of this pattern has led to the creation of the most fragile parts of our projects. The pattern itself has a completely “peaceful” application, the problem is rather that it is put where it fits, and not where it is needed. “If you understand what I mean” (c).
Classification
The pattern exists in several “guises”. Its essence does not change much from this, the danger of its use exists in any of them.
The first, classic view is a certain long-lived object, which accepts another object via the interface, the strategy itself, through the setter, with some change in state.
The second type, a degenerate version of the first - the strategy is adopted once, for the entire life of the object. Those. for one scenario one strategy is used, for another another.
The third type is an executable method, either static or in a short-lived object, which takes an interface strategy as an input parameter. In the “gang of four” this species is called the “Template method”.
The fourth type is interface, aka UI. Sometimes referred to as a “pattern” as a pattern, sometimes as a “container”. On the example of web development, it is a certain piece of markup containing a placeholder (or even more than one) where, during execution, the variable part of the markup that has several different implementations will be rendered. Parallel to the markup, parallel view models or controllers also live in JavaScript code, depending on the architecture adopted in the application, organized in the second form.
Common features of all these cases:
1) some immutable part aka having one implementation, further, I will call it a container
2) mutable part, it’s also a strategy
3) the place of use, creating / invoking the container, determining which strategy the container should use, hereinafter I will call it a script.
Disease development
At first, when a component using this pattern was only implemented in the project, it did not seem so bad. It was applied when it was necessary to create two identical pages (again, for example, web development), which only differ slightly in content in the middle. On the contrary, the developer was glad how beautiful and elegant it was to implement the DRY principle, i.e. completely avoid duplication of code. These are the epithets I heard about a component when it was just created. The one that became the popol of the whole project a few months later.
And since I started to theorize, I’ll go a little further - it is the attempts to implement the DRY principle that, through the strategy pattern, as well as through inheritance, lead to darkness. When, for the sake of DRY, without noticing it, the developer sacrifices the SRP principle, the first and main postulate from SOLID. By the way, DRY is not part of SOLID, and in a conflict, you must sacrifice it, because it does not favor the creation of a stable code, unlike, in fact, SOLID. As it turned out - rather the opposite. Reusing the code should be a pleasant bonus of certain design decisions, and not the purpose of making them.
And the temptation to reuse occurs when the customer comes with a new story to create a third page. After all, she is so similar to the first two. The desire of the customer to realize everything “cheaper” contributes a lot to this, as reusing a previously created container is faster than realizing the page completely. The story came to another developer, who quickly found out that the container’s functionality is not enough, and full refactoring does not fit into the estimates. Another mistake here is that the developer continues to follow the plan set by the estimates, and this happens “in silence”, as it were, there is no responsibility, it lies with the whole team that made such a decision and such an assessment.
And now new functionality is added to the container, new methods and fields are added to the strategy interface. Ifs appear in the container, and in old implementations of the strategy “stubs” appear so as not to break existing pages. At the time of the second story, the component was already doomed, and the further, the worse. It’s becoming more and more difficult for developers to understand how it works, including those that “just recently did something in it”. It's getting harder to make changes. Increasingly, one has to consult with "previous ones" to ask how it works, why some changes were made. It is increasingly likely that even the slightest change will introduce a new defect. Actually, we are already starting to talk about the fact that there is an increasing likelihood of introducing two or more defects, because one defect appears already with a near-unit probability. And now comes the moment when it is impossible to implement the new customer requirements. There are two ways out: either completely rewrite or make an explicit hack. And in the hangar there is just a suitable hack tool - you can emit events from the bottom up, then broadcast from top to bottom when you finished the dirty deals at the top. At the same time, technical debt is no longer increasing; it has already been equal to the cost of selling this component from scratch.
Dry alternative
Inheritance is often criticized, and the corporation of goodness in its Go language generally decided to do without it, and it seems to me that the negative to inheritance partly comes from the experience of implementing the DRY principle through it. The "strategic" DRY also leads to sad results. Remains direct aggregation. To illustrate, I will take a simple example and show how it can be represented as a strategy, that is, a template method and without it.
Suppose we have two very similar scenarios represented by the following pseudo-code:
They repeat 10 lines of X at the beginning and 15 lines of Y at the end. In the middle, one script has lines A, and the other has lines B
СценарийА{
строчка X1
...строчка X10
Строчка А1
...Строчка А5
строчка Y1
...Строчка Y15
}
СценарийВ{
строчка X1
...строчка X10
Строчка B1
...Строчка B3
строчка Y1
...Строчка Y15
}
Option to get rid of duplication through strategy
Контейнер{
строчка X1
...строчка X10
ВызовСтратегии()
строчка Y1
...Строчка Y15
}
СтратегияА{
Строчка А1
...Строчка А5
}
СтратегияВ{
Строчка B1
...Строчка B3
}
СценарийА{
Контейнер(new СтратегияА)
}
СценарийВ{
Контейнер(new СтратегияB)
}
Option through direct aggregation
МетодХ{
строчка X1
...строчка X10
}
МетодY{
строчка Y1
...Строчка Y15
}
МетодА{
Строчка А1
...Строчка А5
}
МетодВ{
Строчка B1
...Строчка B3
}
СценарийА{
МетодХ()
МетодA()
МетодY()
}
СценарийВ{
МетодХ()
МетодB()
МетодY()
}
Here it is assumed that all methods are in different classes.
As I said, at the time of implementation, the first option does not look so bad. Its disadvantage is not that it is initially bad, but that it is unstable to change. And yet, it is worse read, although with a simple example this may not be obvious. When you need to implement the third scenario, which is similar to the first two, but not 100%, there is a desire to reuse the code contained in the container. But it cannot be partially reused, you can only take it in its entirety, so you have to make changes to the container, which immediately carries the risk of breaking other scenarios. The same thing happens when a new requirement involves changes to scenario A, but this should not affect scenario B. In the case of aggregation, method X can easily be replaced by method X 'in one scenario, completely without affecting the others. It is easy to assume that the methods X and X 'can also almost completely coincide, and they can also be subdivided. With a “strategic” approach, if cascaded in the same “strategic” way, then the evil placed in the project is raised to the second degree.
When can
Many examples of the use of the strategy pattern are visible and often used. They are all united by one simple rule - there is no business logic in the container. Absolutely. There may be algorithmic filling, for example a hash table, search or sorting. The strategy contains business logic. The rule according to which one element is equal to another or more or less is business logic. All linq operators are also the embodiment of a pattern, for example .Where () operator is also a template method, and the lambda it accepts is a strategy.
In addition to algorithmic filling, it can be filling connected with the outside world, asynchronous requests, for example, or, in the example of the “gang of four”, a subscription to a mouse click event. What they call callbacks is essentially the same strategy, I hope all my hyper generalizations will forgive me. Also, if it is a UI, then it can be tabs, or a pop-up window.
In a word, it can be anything completely abstracted from business logic.
If you use the strategy pattern in development, and the business logic has got into the container, you should know that you have already crossed the line, and stand ankle-deep in ... mmm, swamp.
Smells
Sometimes it’s not easy to understand where the line is between business logic and general programming tasks. And at first, when the component is just created, to determine that it will bring hemorrhoids in the future is not easy. And if business requirements never change, then this component may never come up. But if there are changes, the following code-smells will inevitably appear:
1. The number of methods passed. The parameter under discussion is not harmful in itself, but it can still hint. Two or three is still fine, but if the strategy contains a dozen methods, then something is probably wrong.
2. Flags. If in addition to methods in the strategy there are fields / properties, it is worth paying attention to what they are called. Fields such as Name, Header, ContentText are valid. But if you see fields such as SkipSomeCheck, IsSomethingAllowed, this means that the strategy is already smelling stinky.
3. Conditional calls. Associated with flags. If the container has a similar code, then you have already gone to the swamp waist-high
if(!strategy.SkipSomeCheck)
{
strategy.CheckSomething().
}
4. Inadequate code. Using an example from JavaScript -
if(strategy.doSomething)
From the name it can be seen that doSomething is a method, but it is checked as a flag. That is, the developers were too lazy to create a flag indicating the type, but used the presence / absence of a method as a flag, and even if it was not called inside the if block. If you encounter this, you should know that the component is already up to its debt in technical debt.
Conclusion
Once again, I want to express my opinion that the root cause of all that I described is not in the pattern as such, but in the fact that it was used for the DRY principle, and this principle was placed above the principle of sole responsibility, like SRP. And, by the way, I have already come across more than once that the principle of sole responsibility is somehow not quite adequately interpreted. About how "my divine class controls a satellite, to control a satellite is his only responsibility." On this note I want to finish my opus and wish less often in response to “why so”, to hear the phrase “historically so happened”.