Consider the following relation between classes:
int main(int, char**) { | class Window { | class Layout { | class Box {
/* Use argc/argv */ | Layout layout; | Box box; | int height,
Window window; | | | max_width;
} | bool print_fps; | public: |
| | Layout(); | public:
| public: | }; | Box (int,int);
| Window (); | | };
| }; | |
I made up this structure just for simplicity, in reality there are many more classes.
In main()
I fetch some application parameters (via configuration files, database, CLI arguments). Now I want to deliver those values to the desired objects.
My question: Which is the best/most elegant way to "break the wall" between the classes so that I can "throw" the configuration and whoever needs it to "grab" it?
Initialy I "opened some doors" and gave the Window
constructor everything that was needed by Window
, Layout
and Box
. Then, Window
gave to Layout
everything needed by Layout
and Box
. And so on.
I quickly realized this was very similar to what Dependency Injection it about but as it turns out, it does not apply directly to my case.
Here I work with primitives like bool
and int
and in fact if I accept them as constructor parameters, I get the result described just above - a very long chain of similar calls: Window(box_height, box_max_width, window_print_fps)
.
What if I'd like to change the type of Box::height
to long
? I would need to walk through every pair of header/source of every class in the chain to change it.
If I want my classes to be isolated (and I do) then Window shouldn't worry about Box and main shouldn't worry about Layout.
Then, my second idea came up: create some JSON-like structure which acts as a config object. Everybody gets a (shared) pointer to it and whenever they want, they say this->config["box"]["height"]
- everyone's happy.
This would kinda work, but there are two problems here: no type safety
and tight coupling between a class (Config
) and the entire code base.
Basically, I see two ways around the problem:
- "Downwards": Out -> In
Objects on the top (outer) care about object in the deep (inner). They push down explicitly what inners want. - "Upwards": In <- Out (the "diagram" is the same, but please wait)
Objects on the bottom (inner) care about themselves on their own. They satisfy their needs by reaching out to some container on the top(outer) and pull what they want.
It's either up or down - I'm trying to think out of the box (In fact, it is a line - just ↑ or ↓) but I ended up only here.
Another issue arising from my two ideas earlier is about the way configuration is parsed:
- If main.cpp (or the config parser) needs to give
int height
toBox
, then it needs to know about box in order to parse the value properly, right? (tight-coupling) If, on the other hand, main.cpp doesn't know about Box (ideally), how should it store the value in a friendly for the box way?
Optional parameters shouldn't be needed in constructors => shouldn't break the application. That is, main should accept the absence of some parameter, but it also needs to know that a setter has to be called for the desired object after it's been constructed with the required parameters.
The whole idea is to strive to these three principles:
- Type safety. Provided by solution 1. but not 2.
- Loose coupling. Provided neither by 1. (main cares about Box) nor by 2. (everybody needs Config)
- Avoid duplication. Provided by 2 but not 1. (many identical parameters forwarded until they reach their target)
I implemented a sub-optimal solution I'll post as a self-answer, which works well for now and it's better than nothing but I'm looking forward for something better!
I decided that I prefer the first approach more, due to the type safety, also I didn't realy like the idea of every every object holding a pointer to some Config.
I did also realize that my perception about DI was not quite correct, or at least that there's a better way to implement it in my case.
So, instead of accepting all parameters of all objects down the tree, a constructor only receives the direct & required dependencies. This entirely solves the duplication issue of 1.
This leaves us only with the tight coupling problem - main has to know about virtually everyone. To know how to create a Window, it needs to create a Layout, and Box, and everybody else.
Firstly, I wanted to take this problem somewhere else, so I made a class to create objects for me. Like a factory, but only used to hide dirty stuff from main.
In a
ConfigFactory
now I store all parameters in some way and pass them when I need to construct an object. It is irrelevant how configuration is stored, I decided to have a simplestruct Data
and used PIMPL so that no recompilation of anyConfigFactory.h
-dependent file is needed when parameter logic is changed (that's supposed to happen quite often)Next, I saw how this guy used templates to make a Factory more generic: https://stackoverflow.com/a/26950454, but instead of working with base and derived classes, pointers and dynamic allocation, I defined a member function which returns a stack-allocated object (ideally):
If there's no specialization of
produce
for the desired type,T
is default constructed - this way it's more flexible when adding/removing specializations.I keep
ConfigFactory.h
only this small and free of any#include
s, so that I do not birng any unneeded dependencies to those who include it.Now, if I need to have something
produced()
, I includeConfigFactory.h
and declare a specialization for it. I put its definition in some source file (ConfigFactory.cpp
) andproduce()
the object using the parameters inthis->data
:main.cpp:
ConfigFactory.cpp, Window.cpp or whoever knows how to make a Window:
produce()
definitions in the corresponding source files, I have some bad feeling about it - because I define things not declared in the header.ConfigFactory.h
for the definitions, it gets quite long and heavy as it needs to know about everyproduce()
able class.Currently I'm using both and in general it works, but it's not perfect.