OOP Layers
Yesterday at work I ran into a pretty nasty bug and while looking through the code I spotted something which even if it wasn't the primary cause of all the mess made it really worst.
The code was going like this :
<?php
class AClassThatDoesSomethingWithThingsAndStuff {
public function aMethodThatAssociatesSeveralThingsWithAStuff(array $things)
{
$aStuff = null;
foreach($things as $thing)
{
$aStuff = $this->aMethodThatUseASomething($thing, $aStuf);
}
}
private function aMethodThatAssociateOneThingWithAStuff(Thing $thing, Stuff $stuff = null)
{
if($stuff == null) {
$stuff = $this->somehowGetAStuff();
}
$thing->associateWithASomething($stuff);
return $stuff;
}
}
?>
As you can see it seems that we always need a Stuff in order to make an association with a Thing. Furthemore we hope to use that same Stuff for all the things.
The main issue with this piece of code is the Stuff we use is going through two layers: the Stiff is created in a sub-layer (in aMethodThatAssociateOneThingWithAStuff
), going up one layer (in `aMethodThatAssociatesSeveralThingsWithAStuff) before returning to the sub-layer again.
An other issue is that we are not really in control of the Stuff used.
I think the code would have been better if it had been written this way :
<?php
class AClassThatDoesSomethingWithThingsAndStuff {
public function aMethodThatAssociatesSeveralThingsWithAStuff(array $things)
{
$stuff = $this->somehowGetAStuff();
foreach($things as $thing)
{
$this->aMethodThatUseASomething($thing, $aStuf);
}
}
private function aMethodThatAssociateOneThingWithAStuff(Thing $thing, Stuff $stuff)
{
$thing->associateWithASomething($stuff);
}
}
?>
This code doesn't break the layers between the methods. The Stuff is get on the top level and passed one level down when needed. We are also able to choose what Stuff we want to pass to the sub-level method.
And it's simpler...
We can even make this way more fun if aMethodThatAssociateOneThingWithAStuff
can throw exceptions.
The code will probably ended up with a try-catch structure inside of the foreach loop. (Our code end up like this at least...)
See below :
<?php
class AClassThatDoesSomethingWithThingsAndStuff {
public function aMethodThatAssociatesSeveralThingsWithAStuff(array $things)
{
$aStuff = null;
foreach($things as $thing)
{
try {
$aStuff = $this->aMethodThatUseASomething($thing, $aStuf);
}
catch(Exception $exception) {
// Do what you want here to treat the exception
}
}
}
private function aMethodThatAssociateOneThingWithAStuff(Thing $thing, Stuff $stuff = null)
{
if($stuff == null) {
$stuff = $this->somehowGetAStuff();
}
$thing->associateWithASomething($stuff);
// Do something else, possibly throwing an exception
return $stuff;
}
}
?>
Imagine the method somehowGetAStuff
has some side effects: writing a new Stuff in the database for instance.
When an exception is thrown the execution enters the catch part of the try-catch and doesn't overwrite the value of $aStuff
.
As long as the method will throw an exception without letting a chance to $aStuff to be initialized
somehowGetAStuff` will be called, with all its side effects...
Breaking layers in OOP can have some really bad consequences and , as you can see, it makes the code harder to read and will probably create some bugs.
This example is based on methods but classes are layers too and thus should not become entangled neither.
As a rule of thumb you should probably be suspicious at a piece of code when you see:
- A test for existence for something passed to your method.
- A memory allocation without desallocation.
- A layer in charge of retrieving something it needs. This is breaking the inversion of dependency principle.
- A method called openSomething without a closeSomething, or the opposite, in the same layer.
because this code is probably running through several layers.
I guess we can make this list way longer.
The header picture is a photography of the Quebrada de Humahuaca in Argentina. The strange look of the mountain, called Cerro de los Siete Colores - The Hill of Seven Colors -, is the result of thousand years of marine sediment. See how beautiful layers can be ?
I took the picture from this website, and I really don't know who I should thanks for it and attribute it to. I'm not sure they do neither...
Hey! I'm on Twitter, if you want to discuss about OOP feel free to come by and say hi! You can comment below as well.
- Improve your automated testing : You will learn how to fix your tests and make them pass from things that slow you down to things that save you time. This is a self-paced video course in French.
- Helping your teams: I help software teams deliver better software sooner. We'll work on technical issues with code, test or architecture, or the process and organization depending on your needs. Book a free call where we'll discuss how things are going on your side and how I can help you.
- Deliver a talk in your organization: I have a few talks that I enjoy presenting, and I can share with your organization(meetup, conference, company, BBL). If you feel that we could work on a new topic together, let's discuss that.