Implementations tightly coupled to their dependencies are difficult to maintain, reuse and test.
In this repository, I will show you 5 ways of refactoring with the intent to "break" tight coupling between business logic and dependencies. A good indicator over the effectiveness of a decoupling is testability: If you are able to test your class and your class only without being influenced by the inner works of the various dependencies, then you know you have done a good decoupling.
Below you will find some tightly coupled code and several ways to break it away from its dependencies. The focus will be mostly on the code, however, the build configuration is often also very important. Being decoupled on the code level but not in the CMake/Bazel/Blueprint files will not get you very far. Last but not least, for every refactored example you will find the unit tests that accompany it.
- Tightly coupled code example
- Dependency Injection (Polymorphic)
- Dependency Injection (Templatized)
- Dependency Injection (Abstract factory)
- Link time switching
- Link time switching (Templatized)
- YouTube
Here is some typical code that is coupled with its dependencies. It will be refactored to demonstrate the different ways you can decouple troublesome code.
class CameraPowerController
{
public:
CameraPowerController(ProductVariant productVariant);
void turnOnCamera();
void turnOffCamera();
private:
std::unique_ptr<AsioSerialPortManager> mAsioSerialPortManager;
};
AsioSerialPortManager
, as its name implies, is a concrete class that utilizes a
3rd-party library.
CameraPowerController::CameraPowerController(ProductVariant productVariant)
{
switch (productVariant)
{
case ProductVariant::A:
mAsioSerialPortManager = std::make_unique<AsioSerialPortManager>(
kSerialDevicePathForVariantA, kBaudRateForVariantA);
return;
case ProductVariant::B:
mAsioSerialPortManager = std::make_unique<AsioSerialPortManager>(
kSerialDevicePathForVariantB, kBaudRateForVariantB);
return;
default:
throw std::logic_error("Unknown variant");
}
}
void CameraPowerController::turnOnCamera()
{
mAsioSerialPortManager->asioWrite("ON");
}
In the CameraPowerController
implementation, we invoke the constructor of AsioSerialPortManager
with some arguments. These arguments are determined based on logic residing within the class. Then,
we call the instantiated class in the other member functions.
Whenever we invoke constructors of other classes, we become get coupled to them. In fact, this is very often a red flag in regards to the testability of the class. Testability is decreased because we become obliged to indirectly test them as well along with the unit we actual want to test. Moreover, compiling and running a binary that involves them may be cumbersome or even impossible in our unit testing environment.
For example, the particular classes may not be compilable for the host platform that runs the
unit tests. This is common when dependencies include resources that are precompiled for a different
operating system or architecture. Additionally, even if the dependant classes are compilable, it
may be impossible to execute them on our host platform, due to them requiring some specific resources
that are only found on a target system. In our case, the AsioSerialPortManager
class needs to
open a connection to a serial device that will only exist on the target we are developing for and not
on the developer's computer.
In other words, if we would like to unit test our CameraPowerController
class and have the unit tests
run on our development machines there is no other option than decoupling it from AsioSerialPortManager
.
The cleanest (IMHO) way to decouple dependencies is to inject their abstractions and use those instead. If such abstractions do not already exist, this means that you have to define an interface (i.e. a pure abstract class) that represents the business value the dependencies offer in a rather generic way. Then, you can have the dependency you need inherit/implement it. Finally, make the constructor of the class under test receive the interface as an argument.
The most straight forward way perhaps is to create an abstract interface directly out of the dependency.
However, this is a bad practice since you will be indirectly coupling your code with the particular implementation.
Instead, I suggest you try to figure out what is the business value in a generic manner. With this in mind,
I created the SerialPortAdapter
interface that generalizes any class that works with the serial port and
exposes a way to transmit data in a generic manner.
struct SerialPortAdapter
{
virtual ~SerialPortAdapter() = default;
virtual void send(std::string_view message) = 0;
};
Then we have the AsioSerialPortAdapter
concrete class that inherits from SerialPortAdapter
and exposes
the AsioSerialPortManager
in a generic manner.
class AsioSerialPortAdapter : public SerialPortAdapter
{
public:
AsioSerialPortAdapter(AsioSerialPortManager* asioSerialPortManager);
void send(std::string_view message) override;
private:
AsioSerialPortManager* mAsioSerialPortManager;
};
AsioSerialPortAdapter::AsioSerialPortAdapter(
AsioSerialPortManager* asioSerialPortManager)
: mAsioSerialPortManager{asioSerialPortManager}
{
}
void AsioSerialPortAdapter::send(std::string_view message)
{
mAsioSerialPortManager->asioWrite(message);
}
In our CameraPowerController
we inject the SerialPortAdapter
. Now, there is nothing specific to the asio
library.
We are effectively decoupled from it.
class CameraPowerController
{
public:
CameraPowerController(SerialPortAdapter* serialPortAdapter);
void turnOnCamera();
void turnOffCamera();
private:
SerialPortAdapter* mSerialPortAdapter;
};
Now that we are injecting things, why not follow the Inversion of Control (IoC) principle all the way?
The serial port configuration is probably not a concern of the CameraPowerController
class, however it takes
place in its constructor. Now that we are injecting resources and letting the users of our classes have control,
we are indirectly encouraged to move these seemingly unrelated functionality outside the class.
std::pair<std::filesystem::path, int>
getAsioSerialPortManagerConfiguration(ProductVariant productVariant)
{
switch (productVariant)
{
case ProductVariant::A:
return std::make_pair(kSerialDevicePathForVariantA,
kBaudRateForVariantA);
case ProductVariant::B:
return std::make_pair(kSerialDevicePathForVariantB,
kBaudRateForVariantB);
default:
throw std::logic_error("Unknown variant");
}
}
int main()
{
const auto [serialDevice, baudRate]
= getAsioSerialPortManagerConfiguration(getProductVariant());
AsioSerialPortManager asioSerialPortManager{serialDevice, baudRate};
AsioSerialPortAdapter asioSerialPortAdapter{&asioSerialPortManager};
CameraPowerController cameraPowerController{&asioSerialPortAdapter};
cameraPowerController.turnOnCamera();
cameraPowerController.turnOffCamera();
return 0;
}
If many levels of indirection and virtual
functions are not desirable (e.g. due to strict requirements on performance)
then injecting your dependencies via the constructor is still the way to go. However, this time you will not use a class hierarchy.
Instead, you can use a class template to determine the type of the constructor argument.
This allows for a looser coupling that is determined by the one that instantiates the class and not the class itself.
template<typename SerialPortManager>
class CameraPowerController
{
public:
CameraPowerController(SerialPortManager* serialPortManager)
: mSerialPortManager{serialPortManager}
{
}
void turnOnCamera()
{
mSerialPortManager->asioWrite("ON");
}
void turnOffCamera()
{
mSerialPortManager->asioWrite("OFF");
}
private:
SerialPortManager* mSerialPortManager;
};
As long as the class template type implements an API equivalent to the AsioSerialPortManager
then our coupling
is only in the design/conceptual level. This is not ideal, however can be good enough for many of the cases. Furthermore,
it should be noted that yet again we have moved the configuration of the serial port outside the class.
If there are really good reasons to give a class control over (some of) its resources then it may not be possible to
instantiate its dependencies in the integration scope (e.g. a main()
function) because they require some information
that resides within the class that uses them.
A good example is the original CameraPowerController
constructor:
CameraPowerController::CameraPowerController(ProductVariant productVariant)
{
switch (productVariant)
{
case ProductVariant::A:
mSerialPortManager = std::make_unique<AsioSerialPortManager>(
kSerialDevicePathForVariantA, kBaudRateForVariantA);
return;
case ProductVariant::B:
mSerialPortManager = std::make_unique<AsioSerialPortManager>(
kSerialDevicePathForVariantB, kBaudRateForVariantB);
return;
default:
throw std::logic_error("Unknown variant");
}
}
AsioSerialPortManager
requires two arguments to be passed to its constructor that are owned by the class that
uses it. To be honest with you, I consider this a design smell. However, if you are really convinced something like this
makes sense, then you can inject an abstract factory class to keep your implementation decoupled from its dependencies.
The abstract factory class takes the arguments that would have otherwise been supplied to the concrete class' constructor and returns an instance of the concrete class. The "trick" is that our class depends on an abstraction returned by the factory function call, not a particular implementation.
class AsioSerialPortManager : public SerialPortManager
{
public:
AsioSerialPortManager(std::filesystem::path serialDevice, int baudRate);
void asioWrite(std::string_view message) override;
private:
asio::io_service mIoService;
asio::serial_port mSerialPort{mIoService};
};
SerialPortManager
is a pure abstract class that generalizes AsioSerialPortManager
.
The SerialPortManagerFactory
pure abstract class "promises" its children to return a child/specialization of the (also abstract) SerialPortManager
.
class AsioSerialPortManagerFactory : public SerialPortManagerFactory
{
public:
std::unique_ptr<SerialPortManager> get(std::filesystem::path serialDevice,
int baudRate) const override;
};
Unsurprisingly, AsioSerialPortManagerFactory
returns an AsioSerialPortManager
instance.
std::unique_ptr<SerialPortManager>
AsioSerialPortManagerFactory::get(std::filesystem::path serialDevice,
int baudRate) const
{
return std::make_unique<AsioSerialPortManager>(serialDevice, baudRate);
}
Our CameraPowerController
class maintains ownership of the resources needed for the instantiation of its dependency but
no longer depends on it. Instead, it depends on the abstraction (i.e. SerialPortManager
) returned by the get
call.
CameraPowerController::CameraPowerController(
SerialPortManagerFactory* serialPortManagerFactory,
ProductVariant productVariant)
{
switch (productVariant)
{
case ProductVariant::A:
mSerialPortManager = serialPortManagerFactory->get(
kSerialDevicePathForVariantA, kBaudRateForVariantA);
return;
case ProductVariant::B:
mSerialPortManager = serialPortManagerFactory->get(
kSerialDevicePathForVariantB, kBaudRateForVariantB);
return;
default:
throw std::logic_error("Unknown variant");
}
}
Occasionally it may not be feasible or practical to refactor existing code but you still wish to break the coupling between the implementation and its dependencies. What you can do, is continue depending on the same declarations during compile time but "replace" the definitions of your dependencies when linking.
Since your code still depends on the same exposed functions and types, you do not need to change it. Instead, when linking you can provide an alternative implementation (e.g. a mock). This time, the "magic" happens on the configuration level.
# CameraPowerController
add_library(camera_power_controller src/CameraPowerController.cpp)
target_include_directories(camera_power_controller PUBLIC include)
target_link_libraries(camera_power_controller
PUBLIC
asio_serial_port_manager
product_variant
)
# src_main
add_executable(src_main main.cpp)
target_link_libraries(src_main
PRIVATE
camera_power_controller
)
On a configuration level our camera_power_controller
target is tightly coupled to the asio_serial_port_manager
one.
No matter what we do on the code level, we will be pulling in the dependency.
target_include_directories(link_switch_camera_power_controller PUBLIC include)
target_link_libraries(link_switch_camera_power_controller
PUBLIC
asio_serial_port_manager_interface
product_variant
)
To break the dependency to the asio_serial_port_manager
target, we instead depend on the
asio_serial_port_manager_interface
which only includes the header files necessary. This means that compilation will work as before.
However, we still need the actual definitions during linking to build a binary. The definitions will be supplied in the integration
scope, i.e. the target that builds the executable.
# src_main
add_executable(link_switch_main link_switch_main.cpp)
target_link_libraries(link_switch_main
PRIVATE
link_switch_camera_power_controller
asio_serial_port_manager
)
The link_switch_main
target, as the one that creates the executable, is responsible for bringing everything together and making
sure that the necessary resources are available for linking. This is eventually the target that depends on asio_serial_port_manager
.
After decoupling, you can provide alternative implementations for the dependencies that will run on different platforms,
e.g. during unit tests. Check out how we would now test CameraPowerController
:
- Test configuration
- Mock
- Alternative
AsioSerialPortManager
implementation that statically invokes the mocks.
Another way to replace dependencies is with a class template. Particularly, turn CameraPowerController
into one.
template<typename SerialPortManager>
class CameraPowerController
{
public:
CameraPowerController(ProductVariant productVariant)
{
switch (productVariant)
{
case ProductVariant::A:
{
const std::filesystem::path kSerialDevicePathForVariantA{
"/dev/CoolCompanyDevice"};
const auto kBaudRateForVariantA = 9600;
mSerialPortManager = std::make_unique<SerialPortManager>(
kSerialDevicePathForVariantA, kBaudRateForVariantA);
}
return;
case ProductVariant::B:
{
const std::filesystem::path kSerialDevicePathForVariantB{"COM3"};
const auto kBaudRateForVariantB = 115200;
mSerialPortManager = std::make_unique<SerialPortManager>(
kSerialDevicePathForVariantB, kBaudRateForVariantB);
}
return;
default:
throw std::logic_error("Unknown variant");
}
}
void turnOffCamera()
{
mSerialPortManager->asioWrite("OFF");
}
private:
std::unique_ptr<SerialPortManager> mSerialPortManager;
};
CameraPowerController
is not coupled to the implementation of AsioSerialPortManager
but something that behaves exactly
like one. This means we can create variants by using a different type during the CameraPowerController
instantiation.
For the unit tests, we follow a similar approach to the other "link switch" method.
This tutorial also exists as a video on YouTube. Check it out and if you enjoyed it don't forget to comment, like and subscribe! 😊