The coding blog of Alastair Smith, a software developer based in Cambridge, UK. Interested in DevOps, Azure, Kubernetes, .NET Core, and VueJS.
Last time, I rounded off the series-within-a-series on class design and usage, Working Classes. The next topic for dissection is routine design, creation and usage, and this topic will be handled over two posts. This post will form a bit of an introduction and cover design considerations for high-quality routines through the classification of different kinds of routine cohesion. So, without further ado, let’s get cracking! A good place to start is to define what a routine is, and McConnell describes it as an individual method or procedure invokable for a single purpose. Note that implicit in this definition is an assumption that the Single Responsibility Principle applies to methods as well as classes.
McConnell then provides an example of a low quality routine, and suggests to the reader that they pick it apart and find as many errors in it as they can. I’ve included my opinions and then McConnell’s answers below; don’t scroll too far if you want to try this exercise for yourself. The routine is as follows (as printed in Code Complete, 2nd Edition):
I picked out the following things as being bad practice or otherwise wrong:
HandleStuff
is a very vague name that gives no hints as to what the routine is supposed to achieve.switch
ing on an int, then this provides a more readable alternative to sequence of if..else if
Here’s McConnell’s list of answers:
So, with that in mind, what are the valid reasons to create a routine? Well, as we saw last time with creating classes, the single most important reason to create a routine is to reduce complexity. Again, this is an effective technique for hiding information so you don’t have to juggle that along with the rest of the routine. The deep nesting of an inner loop or conditional is a prime target for the Extract Method refactoring.
Routines are a great tool for managing duplication, and indeed avoiding it in the first place. If reducing complexity is the most important reason to create a routine, avoiding duplicate code is the most popular. If you find yourself creating similar code in different routines, then you likely have an error in your decomposition: at this point, you can pull out the duplicate code, place a more generic version in a common base class, and move the specialised routines into subclasses; alternatively, you can move the common code into its own routine and let both methods call the new routine. Furthermore, this saves space used by the duplicated code, making modifications easier because you only have to update one place, and making your code more reliable as there is only one place to check to ensure the code is correct.
[img_assist | nid=139 | title= | desc=Photo by José Encarnação | link=none | align=center | width=400 | height=300] |
You can create new routines to support subclassing, as well. Less code is needed to override a short, well-factored routine than a long and sprawling routine, because the original routine is clearly-defined and well-understood. As a result, simple overridable routines can help reduce the chance of error in the subclass implementations.
In any program, it’s sensible to hide the order in which events happen to be processed from the wider world, because it shouldn’t matter beyond the scope of the routine. In a blog application, for example, you don’t really want the code dealing with saving data to know that a connection to the database must be set up and opened, and an SQL statement prepared and populated with data, before the data can actually be saved to the database. Instead, you can hide all this away in your Save()
method, and ideally it would be factored out into separate routines on a dedicated class such as a Repository.
If languages like C++ are your thing, routines can be used effectively to manage the pointer operations integral to these languages. This technique is less appropriate to modern languages that abstract the pointer operations away from the developer’s eyes, but it allows the developer to concentrate on the intent of the operation rather than its mechanics. A similar argument holds for managing the portability of your system: using routines allows you isolate the non-portable capabilities of your system, such as database-specific functionality, from the rest of your system. Such issues may not be applicable to your system, however.
One of my personal favourite reasons to create a new routine is to simplify a complicated boolean test. Understanding all the ins and outs of complicated boolean conditions is rarely necessary, and inlining them in your code presents a barrier to readability. It’s much easier to grok a complicated condition if it’s wrapped in a method with a good, descriptive name: the finer details of the test are abstracted away and summarised nicely. Furthermore, this emphasises the test’s significance, encouraging your to expend extra effort to make the detail of the test clear and readable inside the function. This is a technique I’ve used at work a few times recently, in both my own code and in code reviews, and it seems to be paying off well.
Finally, you can introduce new routines to improve performance. This is because it is easier to optimise the code in one place instead of several, and makes your code easier to profile.
You should never introduce a new routine to ensure all routines are small. This is a bit of an anti-pattern, and leads to arbitrary divisions of labour. This is then usually compounded by the fact that such routines are difficult to name. You very quickly end up with a mess.
The biggest mental block to creating effective routines is a reluctance to create a simple routine for a simple purpose. However, these make code more self-documenting. Small routines can turn into larger routines, once newly-discovered errors are accounted for. Furthermore, many of the reasons to create a class are also good reasons to create a routine:
Cohesion, introduced by Wayne Stevens, Glenford Myers and Larry Constantine in 1974, is the workhorse design heuristic at the routine level. Cohesion refers to how closely the operations within a routine are related, and can also be referred to as “strength”. The goal is to achieve maximum cohesion: each routine does one thing well and nothing else; this is known as functional cohesion. The following forms of cohesion are considered poor, and prime for refactoring.
[img_assist | nid=140 | title=Cohesion | desc=Photo by Yannig Van de Wouwer | link=none | url=http://www.flickr.com/photos/nogood/211866952/ | align=center | width=399 | height=266] |
Sequential cohesion exists when a routine contains operations that must be performed in a specific order, sharing data from step to step, and don’t make up a complete function when done together. For example, in a routine that calculates age and time until retirement from a birth date, if the time to retirement is calculated from the result of the age calculation, the routine is considered sequentially cohesive. To make it functionally cohesive, create separate routines where each takes the birth date as a parameter. Time-to-retirement could call the Age routine and each would still have functional cohesion.
Communicational cohesion is defined by operations in a routine that use the same data, but are otherwise unrelated. For example, a given summary-printing routine also reinitialises the summary data; the operations can be split into separate routines, with the summary reinitialisation routine being called from the location where the summary is created in the first place.
Temporal cohesion occurs when operations are combined into a routine because they’re all executed together. Startup()
and Shutdown()
methods are prime examples of temporal cohesion. Functional cohesion can be achieved by factoring out the individual jobs of the routine to separate routines and calling them from the original routine.
The following kinds of cohesion are generally unacceptable, and it is considered better to put in the effort to re-write them rather than put up with code that is poorly organised, hard to debug and hard to modify.
Procedural cohesion occurs when operations are done in a specified order (but don’t necessarily share data). A set of operations is organised into a specified order and the operations don’t need to be combined for any other reason; a routine that gets an employee name, then an address, then a phone number would be considered procedurally cohesive. To make the routine functionally cohesive, put the separate operations into their own routines, and make sure the calling routine has a single complete job (e.g., GetEmployee()).
Logical cohesion occurs when several operations are placed into a single routine, and one of the operations is selected by a control flag passed in as a parameter; i.e., only the control flow ties the operations together: they’re otherwise separated out in a big if or switch statement together and the operations themselves are unrelated. For example, a method called InputAll() can input customer names, timesheet info, or inventory data, depending on the flag passed in. This can be improved by separating the operations into their own routines; if there is shared code between the operations, this is moved into a helper method and the operations are packaged into a class. There is an exception, however: a routine containing nothing but a series of if or switch statements and calls to other routines. McConnell terms this an “event handler”, which is potentially a confusing term depending on your background.
Coincidental cohesion occurs when operations in a routine have no discernible relationship, and is also known as “no cohesion” or “chaotic cohesion”. Unfortunately this is not easily improved upon, and if you find yourself faced with a coincidentally cohesive routine, the only practical approach is to give it a deeper redesign and reimplementation.
This post provided an introduction to key considerations for routines, including when to create new routines, when not to create new routines, and the concept of routine cohesion. Part two will cover naming conventions for routines, and the use of routine parameters.