-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy path2019-08-03-ioc-techniques.html
413 lines (322 loc) · 27.9 KB
/
2019-08-03-ioc-techniques.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
<!DOCTYPE html>
<html lang="en">
<head>
<title>
Three Techniques for Inverting Control, in Python
</title>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="author" content="Harry Percival and Bob Gregory">
<meta name="description" content="">
<meta property="og:title" content="cosmic_python" />
<meta property="og:type" content="website" />
<meta property="og:url" content="https://www.cosmicpython.com/blog/2019-08-03-ioc-techniques.html" />
<meta property="og:image" content="https://www.cosmicpython.com/images/lobster_nebula.jpg" />
<link rel="stylesheet" href="//fonts.googleapis.com/css?family=Roboto:300,300italic,700,700italic">
<link rel="stylesheet" href="//cdnjs.cloudflare.com/ajax/libs/normalize/5.0.0/normalize.css">
<link rel="stylesheet" href="//cdnjs.cloudflare.com/ajax/libs/milligram/1.3.0/milligram.css">
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/[email protected]/friendly.css">
<link rel="stylesheet" href="/styles.css">
</head>
<body>
<main class="wrapper">
<nav class="navigation">
<section class="container">
<a class="navigation-title" href="/">
<h1 class="title">Cosmic Python</h1>
</a>
</section>
</nav>
<section class="container">
<h1> Three Techniques for Inverting Control, in Python</h1>
<p>by David, 2019-08-03</p>
<div class="row">
<div class="column">
<img src="/images/ioc-techniques.jpg" />
</div>
</div>
<div class="content">
<p><em>David was a tech reviewer for the <a href="/">book</a> and these two excellent
articles on inversion of control are cross-posted from
<a href="https://seddonym.me/blog/">his blog where you can find lots more excellent content</a>.</em></p>
<p>In <a href="/blog/2019-04-15-inversion-of-control.html">the previous post</a> we learned how Inversion of Control can
be visualised as follows:</p>
<p><img src="/images/ioc-techniques/a-b-plugin.png" alt="B plugging into A"></p>
<p><code>B</code> plugs into <code>A</code>. <code>A</code> provides a mechanism for <code>B</code> to do this — but otherwise <code>A</code> need know nothing about <code>B</code>.</p>
<p>The diagram provides a high level view of the mechanism, but how is this actually implemented?</p>
<h2>A pattern for inverting control</h2>
<p>Getting a little closer to the code structure, we can use this powerful pattern:</p>
<p><img src="/images/ioc-techniques/di-pattern.png" alt="main pointing to A and B, A pointing to <B>, B pointing (open arrow) to <B>"></p>
<p>This is the basic shape of inversion of control. Captured within the notation, which may or may not be familiar
to you, are the concepts of <em>abstraction</em>, <em>implementation</em> and <em>interface</em>. These concepts are all important
to understanding the techniques we’ll be employing. Let’s make sure we understand what they mean when applied
to Python.</p>
<h3>Abstractions, implementations and interfaces — in Python</h3>
<p>Consider three Python classes:</p>
<div class="codehilite"><pre><span></span><code><span class="k">class</span> <span class="nc">Animal</span><span class="p">:</span>
<span class="k">def</span> <span class="nf">speak</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
<span class="k">raise</span> <span class="ne">NotImplementedError</span>
<span class="k">class</span> <span class="nc">Cat</span><span class="p">(</span><span class="n">Animal</span><span class="p">):</span>
<span class="k">def</span> <span class="nf">speak</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Meow."</span><span class="p">)</span>
<span class="k">class</span> <span class="nc">Dog</span><span class="p">(</span><span class="n">Animal</span><span class="p">):</span>
<span class="k">def</span> <span class="nf">speak</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Woof."</span><span class="p">)</span>
</code></pre></div>
<p>In this example, <code>Animal</code> is an <em>abstraction</em>: it declares its <code>speak</code> method, but it’s not intended to be run (as
is signalled by the <code>NotImplementedError</code>).</p>
<p><code>Cat</code> and <code>Dog</code>, however, are <em>implementations</em>: they both implement the <code>speak</code> method, each in their own way.</p>
<p>The <code>speak</code> method can be thought of as an <em>interface</em>: a common way in which other code may interact with
these classes.</p>
<p>This relationship of classes is often drawn like this, with an open arrow indicating that <code>Cat</code> and <code>Dog</code> are concrete
implementations of <code>Animal</code>.</p>
<p><img src="/images/ioc-techniques/animal-cat-dog.png" alt="Diagram of Cat and Dog subclassing Animal" /></p>
<h4>Polymorphism and duck typing</h4>
<p>Because <code>Cat</code> and <code>Dog</code> implement a shared interface, we can interact with either class without knowing which one it is:</p>
<div class="codehilite"><pre><span></span><code><span class="k">def</span> <span class="nf">make_animal_speak</span><span class="p">(</span><span class="n">animal</span><span class="p">):</span>
<span class="n">animal</span><span class="o">.</span><span class="n">speak</span><span class="p">()</span>
<span class="n">make_animal_speak</span><span class="p">(</span><span class="n">Cat</span><span class="p">())</span>
<span class="n">make_animal_speak</span><span class="p">(</span><span class="n">Dog</span><span class="p">())</span>
</code></pre></div>
<p>The <code>make_animal_speak</code> function need not know anything about cats or dogs; all it has to know is how to interact
with the abstract concept of an animal. Interacting with objects without knowing
their specific type, only their interface, is known as ‘polymorphism’.</p>
<p>Of course, in Python we don’t actually <em>need</em> the base class:</p>
<div class="codehilite"><pre><span></span><code><span class="k">class</span> <span class="nc">Cat</span><span class="p">:</span>
<span class="k">def</span> <span class="nf">speak</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Meow."</span><span class="p">)</span>
<span class="k">class</span> <span class="nc">Dog</span><span class="p">:</span>
<span class="k">def</span> <span class="nf">speak</span><span class="p">(</span><span class="bp">self</span><span class="p">):</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Woof."</span><span class="p">)</span>
</code></pre></div>
<p>Even if <code>Cat</code> and <code>Dog</code> don’t inherit <code>Animal</code>, they can still be passed to <code>make_animal_speak</code> and things
will work just fine. This informal ability to interact with an object without it explicitly declaring an interface
is known as ‘duck typing’.</p>
<p>We aren’t limited to classes; functions may also be used in this way:</p>
<div class="codehilite"><pre><span></span><code><span class="k">def</span> <span class="nf">notify_by_email</span><span class="p">(</span><span class="n">customer</span><span class="p">,</span> <span class="n">event</span><span class="p">):</span>
<span class="o">...</span>
<span class="k">def</span> <span class="nf">notify_by_text_message</span><span class="p">(</span><span class="n">customer</span><span class="p">,</span> <span class="n">event</span><span class="p">):</span>
<span class="o">...</span>
<span class="k">for</span> <span class="n">notify</span> <span class="ow">in</span> <span class="p">(</span><span class="n">notify_by_email</span><span class="p">,</span> <span class="n">notify_by_text_message</span><span class="p">):</span>
<span class="n">notify</span><span class="p">(</span><span class="n">customer</span><span class="p">,</span> <span class="n">event</span><span class="p">)</span>
</code></pre></div>
<p>We may even use Python modules:</p>
<div class="codehilite"><pre><span></span><code><span class="kn">import</span> <span class="nn">email</span>
<span class="kn">import</span> <span class="nn">text_message</span>
<span class="k">for</span> <span class="n">notification_method</span> <span class="ow">in</span> <span class="p">(</span><span class="n">email</span><span class="p">,</span> <span class="n">text_message</span><span class="p">):</span>
<span class="n">notification_method</span><span class="o">.</span><span class="n">notify</span><span class="p">(</span><span class="n">customer</span><span class="p">,</span> <span class="n">event</span><span class="p">)</span>
</code></pre></div>
<p>Whether a shared interface is manifested in a formal, object oriented manner, or more implicitly, we can
generalise the separation between the interface and the implementation like so:</p>
<p><img src="/images/ioc-techniques/interface-implementation.png" alt="Diagram of implementation inheriting abstract interface" /></p>
<p>This separation will give us a lot of power, as we’ll see now.</p>
<h2>A second look at the pattern</h2>
<p>Let’s look again at the Inversion of Control pattern.</p>
<p><img src="/images/ioc-techniques/di-pattern.png" alt="main pointing to A and B, A pointing to <B>, B pointing (open arrow) to <B>" /></p>
<p>In order to invert control between <code>A</code> and <code>B</code>, we’ve added two things to our design.</p>
<p>The first is <code><<B>></code>. We’ve separated out into its abstraction (which <code>A</code> will continue to depend on and know about),
from its implementation (of which <code>A</code> is blissfully ignorant).</p>
<p>However, somehow the software will need to make sure that <code>B</code> is used in place of its abstraction. We therefore need
some orchestration code that knows about both <code>A</code> and <code>B</code>, and does the final linking of them together. I’ve called
this <code>main</code>.</p>
<p>It’s now time to look at the techniques we may use for doing this.</p>
<h2>Technique One: Dependency Injection</h2>
<p>Dependency Injection is where a piece of code allows the calling code to control its dependencies.</p>
<p>Let’s begin with the following function, which doesn’t yet support dependency injection:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># hello_world.py</span>
<span class="k">def</span> <span class="nf">hello_world</span><span class="p">():</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Hello, world."</span><span class="p">)</span>
</code></pre></div>
<p>This function is called from a top level function like so:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># main.py</span>
<span class="kn">from</span> <span class="nn">hello_world</span> <span class="kn">import</span> <span class="n">hello_world</span>
<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span>
<span class="n">hello_world</span><span class="p">()</span>
</code></pre></div>
<p><code>hello_world</code> has one dependency that is of interest to us: the built in function <code>print</code>. We can draw a diagram
of these dependencies like this:</p>
<p><img src="/images/ioc-techniques/main-hw-print.png" alt="Main pointing to hello_world pointing to print" /></p>
<p>The first step is to identify the abstraction that <code>print</code> implements. We could think of this simply as a
function that outputs a message it is supplied — let’s call it <code>output_function</code>.</p>
<p>Now, we adjust <code>hello_world</code> so it supports the injection of the implementation of <code>output_function</code>. Drum roll please…</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># hello_world.py</span>
<span class="k">def</span> <span class="nf">hello_world</span><span class="p">(</span><span class="n">output_function</span><span class="p">):</span>
<span class="n">output_function</span><span class="p">(</span><span class="s2">"Hello, world."</span><span class="p">)</span>
</code></pre></div>
<p>All we do is allow it to receive the output function as an argument. The orchestration code then passes in the <code>print</code> function via the argument:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># main.py</span>
<span class="kn">import</span> <span class="nn">hello_world</span>
<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">hello_world</span><span class="p">(</span><span class="n">output_function</span><span class="o">=</span><span class="nb">print</span><span class="p">)</span>
</code></pre></div>
<p>That’s it. It couldn’t get much simpler, could it? In this example, we’re injecting a callable, but other
implementations could expect a class, an instance or even a module.</p>
<p>With very little code, we have moved the dependency out of <code>hello_world</code>, into the top level function:</p>
<p><img src="/images/ioc-techniques/main-hw-print-output.png" alt="Main pointing to hello_world and print, hello_world pointing to <output>, print pointing (open arrow) to <output>." /></p>
<p>Notice that although there isn’t a formally declared abstract <code>output_function</code>, that concept is implicitly there, so
I’ve included it in the diagram.</p>
<h2>Technique Two: Registry</h2>
<p>A <em>Registry</em> is a store that one piece of code reads from to decide how to behave, which may be
written to by other parts of the system. Registries require a bit more machinery that dependency injection.</p>
<p>They take two forms: <em>Configuration</em> and <em>Subscriber</em>:</p>
<h3>The Configuration Registry</h3>
<p>A configuration registry gets populated once, and only once. A piece of code uses one
to allow its behaviour to be configured from outside.</p>
<p>Although this needs more machinery than dependency injection, it doesn’t need much:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># hello_world.py</span>
<span class="n">config</span> <span class="o">=</span> <span class="p">{}</span>
<span class="k">def</span> <span class="nf">hello_world</span><span class="p">():</span>
<span class="n">output_function</span> <span class="o">=</span> <span class="n">config</span><span class="p">[</span><span class="s2">"OUTPUT_FUNCTION"</span><span class="p">]</span>
<span class="n">output_function</span><span class="p">(</span><span class="s2">"Hello, world."</span><span class="p">)</span>
</code></pre></div>
<p>To complete the picture, here’s how it could be configured externally:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># main.py</span>
<span class="kn">import</span> <span class="nn">hello_world</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">config</span><span class="p">[</span><span class="s2">"OUTPUT_FUNCTION"</span><span class="p">]</span> <span class="o">=</span> <span class="nb">print</span>
<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">hello_world</span><span class="p">()</span>
</code></pre></div>
<p>The machinery in this case is simply a dictionary that is written to from outside the module. In a real world system,
we might want a slightly more sophisticated config system (making it immutable for example, is a good idea). But at heart,
any key-value store will do.</p>
<p>As with dependency injection, the output function’s implementation has been lifted out, so <code>hello_world</code> no longer depends on it.</p>
<p><img src="/images/ioc-techniques/configuration-registry.png" alt="Configuration registry" /></p>
<h3>The Subscriber Registry</h3>
<p>In contrast to a configuration registry, which should only be populated once, a
subscriber registry may be populated an arbitrary number of times by different parts
of the system.</p>
<p>Let’s develop our ultra-trivial example to use this pattern. Instead of saying “Hello, world”, we want
to greet an arbitrary number of people: “Hello, John.”, “Hello, Martha.”, etc. Other parts of the system should be
able to add people to the list of those we should greet.</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># hello_people.py</span>
<span class="n">people</span> <span class="o">=</span> <span class="p">[]</span>
<span class="k">def</span> <span class="nf">hello_people</span><span class="p">():</span>
<span class="k">for</span> <span class="n">person</span> <span class="ow">in</span> <span class="n">people</span><span class="p">:</span>
<span class="nb">print</span><span class="p">(</span><span class="sa">f</span><span class="s2">"Hello, </span><span class="si">{</span><span class="n">person</span><span class="si">}</span><span class="s2">."</span><span class="p">)</span>
</code></pre></div>
<div class="codehilite"><pre><span></span><code><span class="c1"># john.py</span>
<span class="kn">import</span> <span class="nn">hello_people</span>
<span class="n">hello_people</span><span class="o">.</span><span class="n">people</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="s2">"John"</span><span class="p">)</span>
</code></pre></div>
<div class="codehilite"><pre><span></span><code><span class="c1"># martha.py</span>
<span class="kn">import</span> <span class="nn">hello_people</span>
<span class="n">hello_people</span><span class="o">.</span><span class="n">people</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="s2">"Martha"</span><span class="p">)</span>
</code></pre></div>
<p>As with the configuration registry, there is a store that can be written to from outside. But instead of
being a dictionary, it’s a list. This list is populated, typically
at startup, by other components scattered throughout the system. When the time is right,
the code works through each item one by one.</p>
<p>A diagram of this system would be:</p>
<p><img src="/images/ioc-techniques/subscriber-registry.png" alt="Subscriber registry" /></p>
<p>Notice that in this case, <code>main</code> doesn’t need to know about the registry — instead, it’s the subscribers elsewhere
in the system that write to it.</p>
<h4>Subscribing to events</h4>
<p>A common reason for using a subscriber registry is to allow other parts of a system to react to events
that happen one place, without that place directly calling them. This is often solved by the <a href="https://sourcemaking.com/design_patterns/observer">Observer Pattern</a>,
a.k.a. pub/sub.</p>
<p>We may implement this in much the same way as above, except instead of adding strings to a list, we add callables:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># hello_world.py</span>
<span class="n">subscribers</span> <span class="o">=</span> <span class="p">[]</span>
<span class="k">def</span> <span class="nf">hello_world</span><span class="p">():</span>
<span class="nb">print</span><span class="p">(</span><span class="s2">"Hello, world."</span><span class="p">)</span>
<span class="k">for</span> <span class="n">subscriber</span> <span class="ow">in</span> <span class="n">subscribers</span><span class="p">:</span>
<span class="n">subscriber</span><span class="p">()</span>
</code></pre></div>
<div class="codehilite"><pre><span></span><code><span class="c1"># log.py</span>
<span class="kn">import</span> <span class="nn">hello_world</span>
<span class="k">def</span> <span class="nf">write_to_log</span><span class="p">():</span>
<span class="o">...</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">subscribers</span><span class="o">.</span><span class="n">append</span><span class="p">(</span><span class="n">write_to_log</span><span class="p">)</span>
</code></pre></div>
<h2>Technique Three: Monkey Patching</h2>
<p>Our final technique, <em>Monkey Patching</em>, is very different to the others, as it doesn’t use the Inversion of Control
pattern described above.</p>
<p>If our <code>hello_world</code> function doesn’t implement any hooks for injecting its output function, we <em>could</em> monkey patch the
built in <code>print</code> function with something different:</p>
<div class="codehilite"><pre><span></span><code><span class="c1"># main.py</span>
<span class="kn">import</span> <span class="nn">hello_world</span>
<span class="kn">from</span> <span class="nn">print_twice</span> <span class="kn">import</span> <span class="n">print_twice</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">print</span> <span class="o">=</span> <span class="n">print_twice</span>
<span class="k">if</span> <span class="vm">__name__</span> <span class="o">==</span> <span class="s2">"__main__"</span><span class="p">:</span>
<span class="n">hello_world</span><span class="o">.</span><span class="n">hello_world</span><span class="p">()</span>
</code></pre></div>
<p>Monkey patching takes other forms. You could manipulate to your heart’s content some hapless class defined elsewhere
— changing attributes, swapping in other methods, and generally doing whatever you like to it.</p>
<h2>Choosing a technique</h2>
<p>Given these three techniques, which should you choose, and when?</p>
<h3>When to use monkey patching</h3>
<p>Code that abuses the Python’s dynamic power can be extremely
difficult to understand or maintain. The problem is that if you are reading monkey patched code, you have no clue
to tell you that it is being manipulated elsewhere.</p>
<p>Monkey patching should be reserved for desperate times, where you don’t have the ability to change the code you’re
patching, and it’s really, truly impractical to do anything else.</p>
<p>Instead of monkey patching, it’s much better to use one of the other inversion of control techniques.
These expose an API that formally provides the hooks that other code can use to change behaviour, which is easier
to reason about and predict.</p>
<p>A legitimate exception is testing, where you can make use of <code>unittest.mock.patch</code>. This <em>is</em> monkey patching, but it’s
a pragmatic way to manipulate dependencies when testing code. Even then, some people view testing like this as
a code smell.</p>
<h3>When to use dependency injection</h3>
<p>If your dependencies change at runtime, you’ll need dependency injection. Its alternative, the registry,
is best kept immutable. You don’t want to be changing what’s in a registry, except at application start up.</p>
<p><a href="https://docs.python.org/3/library/json.html"><code>json.dumps</code></a> is a good example from the standard library which uses
dependency injection. It serializes a Python object to a JSON string, but if the default encoding doesn’t support what
you’re trying to serialize, it allows you to pass in a custom encoder class.</p>
<p>Even if you don’t need dependencies to change, dependency injection is a good technique if you want a really simple way
of overriding dependencies, and don’t want the extra machinery of configuration.</p>
<p>However, if you are having to inject the same dependency a lot, you might find your code becomes rather unwieldy and
repetitive. This can also happen if you only need the dependency quite deep in the call stack, and are having to pass
it around a lot of functions.</p>
<h3>When to use registries</h3>
<p>Registries are a good choice if the dependency can be fixed at start up time. While you could use dependency injection
instead, the registry is a good way to keep configuration separate from the control flow code.</p>
<p>Use a configuration registry when you need something configured to a single value. If there is already a
configuration system in place (e.g. if you’re using a framework that has a way of providing global configuration) then
there’s even less extra machinery to set up. A good example of this is Django’s ORM, which provides a Python API around different database engines. The ORM does not depend on any one database engine; instead,
you <a href="https://docs.djangoproject.com/en/2.2/ref/settings/#databases">configure your project to use a particular database engine</a>
via Django’s configuration system. </p>
<p>Use a subscriber registry for pub/sub, or when you depend on an arbitrary number of values. Django <a href="https://docs.djangoproject.com/en/2.2/topics/signals/">signals</a>,
which are a pub/sub mechanism, use this pattern. A rather different use case, also from Django,
is its <a href="https://docs.djangoproject.com/en/2.2/ref/contrib/admin/">admin site</a>. This uses a subscriber registry to
allow different database tables to be registered with it, exposing a CRUD interface in the UI.</p>
<p>Configuration registries <em>may</em> be used in place of subscriber registries for configuring,
say, a list — if you prefer doing your linking up in single place, rather than scattering it throughout the application.</p>
<h2>Conclusion</h2>
<p>I hope these examples, which were as simple as I could think of, have shown how easy it is to invert control in Python.
While it’s not always the most obvious way to structure things, it can be achieved with very little extra code.</p>
<p>In the real world, you may prefer to employ these techniques with a bit more structure. I often choose classes rather
than functions as the swappable dependencies, as they allow you to declare the interface in a more formal way.
Dependency injection, too, has more sophisticated implementations, and there are even some third party frameworks available.</p>
<p>There are costs as well as benefits. Locally, code that employs IoC may be harder to understand and debug, so be sure that it
is reducing complication overall.</p>
<p>Whichever approaches you take, the important thing to remember is that the relationship of dependencies in a software package is
crucial to how easy it will be to understand and change. Following the path of least resistance can result in dependencies
being structured in ways that are, in fact, unnecessarily difficult to work with. These techniques give you the power
to invert dependencies where appropriate, allowing you to create more maintainable, modular code. Use them wisely!</p>
</div>
<div id="disqus_thread" style="margin: 10px"></div>
<script>
var disqus_config = function () {
this.page.url = 'https://www.cosmicpython.com/blog/2019-08-03-ioc-techniques.html';
this.page.identifier = 'cosmic-python--blog-2019-08-03-ioc-techniques';
};
(function() { // DON'T EDIT BELOW THIS LINE
var d = document, s = d.createElement('script');
s.src = 'https://cosmic-python.disqus.com/embed.js';
s.setAttribute('data-timestamp', +new Date());
(d.head || d.body).appendChild(s);
})();
</script>
<noscript>Please enable JavaScript to view the <a href="https://disqus.com/?ref_noscript">comments powered by Disqus.</a></noscript>
<!-- Global site tag (gtag.js) - Google Analytics -->
<script async src="https://www.googletagmanager.com/gtag/js?id=UA-40928035-3"></script>
<script>
window.dataLayer = window.dataLayer || [];
function gtag(){dataLayer.push(arguments);}
gtag('js', new Date());
gtag('config', 'UA-40928035-3');
</script>
</section>
</main>
</body>
</html>