Sunday, 5 June 2016

Abstract Defence



The proper application of abstraction is potentially the most important skill an engineer can develop.
When correctly applied abstraction leads to loose coupling which in turns increases the possibilities for you code base.
However abstraction sometimes gets a bad name and is blamed for adding complexity, while this can be true when its badly applied we shouldn't let this blind us to the immense benefits it can bring.
Bad Code is Bad Code
The main criticisms of abstraction are related to it taking more time to implement and it adding more complexity.
No matter how your approach the problem you have to write code to solve it, by introducing abstraction all you are doing is taking time to think about the interface to that solution.
Thinking about how your code is used should not be considered an onerous task, time saved by not thinking about things is paid back in full when unintended consequences are encountered further down the line.
When the use of abstraction adds complication this is usually either because the architecture and implementation of the code is already complex or because abstraction is being badly applied.
Quite often this is because the abstraction won't hold up over time, the most common cause of this being the abstraction leaks implementation detail. Rather than being constructed from the point of view of the caller it screams details about what is happening under the hood.
When these details change the abstraction breaks and people get confused.
Why Abstract?
So how do we make sure we do apply abstraction properly, firstly why are we applying abstraction:
  • To reduce the surface area of change.
  • To control the interface to functionality.
  • Testability.
All three could be summed up by saying we want to remove dependency on detail.
Details change and we therefore want to reduce the code that gets re-written when the change occurs.
When we interact with a piece of code we don't want to interact with the detail we want to interact with the functionality.
When I'm testing my code I don't want to be testing the detail of a dependency.
If there is no detail we don't need abstraction, a class modelling a customer contains no implementation detail, its purely data driven with no secrets to hide. 
Where ever you see detail think about the functionality being offered that could be represented by an abstraction.
What to Abstract?
If we are to achieve this de-coupling of detail what are the kinds of things that we should be abstracting:
  • Technology.
  • Code we didn't write.
  • Code we might lift and shift.
The technology being used to save data, send data or work with data is detail with a high propensity for change and also has very little to do with the functionality being offered.
Its bad enough when change in code you wrote causes widespread change in your code base, the impact is felt even more when change in code written by a 3rd party causes the same disruption. In this instance you not only don't control the change you don't control its impact either.
Obviously a dose of realism is required here, I'm not suggesting you try and abstract yourself from the OS your code is running on. Instead you need to make a call, can I realistically protect myself from these guys making changes, sometime the answer will be yes sometimes no.
Finally as I've said many times, a good developer is a lazy developer who is always looking for opportunities to not write more code.
Code re-use is a much easier prospect when the wiring and plumbing aren't exposed. When you have a clearly defined junction between code requiring functionality and code providing it you will find all sorts of new ways to write once and run everywhere.
As with many techniques in development over using abstraction will degrade a code base, however the opportunities for abstraction to improve your code base are numerous and plentiful you just need to develop a keen eye for spotting them.

No comments:

Post a Comment