Clean Code | Chapter(3) | Functions
The following rules is about the mechanics of writing functions well. If you follow these rules, your functions will be short, well named, and nicely organized.
1) Small:
Listing 3-1 | HtmlUtil.java
Listing 3-2 | HtmlUtil.java (refactored)
Listing 3-3 | HtmlUtil.java (re-refactored)
2) Do One Thing:
TO "RenderPageWithSetupsAndTeardowns", we check to see whether the page is a test page and if so, we include the setups and teardowns. In either case we render the page in HTML.
3) One Level of Abstraction per Function:
Make your code reads like a top-down story (top-down set of TO paragraphs). To do that, should every function written in a single level of abstraction and followed by the next function in the next level of abstraction, and the third one in the third level of abstraction, and so on. For example, the following snippet.
Listing 3-7 SetupTeardownIncluder.java
4) Switch Statements:
The problem with the above function is that there are an unlimited number of other functions that will have the same switch structure, and this repetition is an indication that your code not well design. For example we could have isPayday and deliverPay as the following,
and may be there are a lot of other functions and also have the same switch structure.
The solution (See the following nippet) to this problem is to insert the switch statement in the ABSTRACT FACTORY, and never let anyone see it. The factory will use the switch statement to create appropriate instances of the derivatives of Employee, and the various functions, such as calculatePay, isPayday, and deliverPay, will be dispatched polymorphically through the Employee interface.
So, by using the new design we can make sure that the switch statement is inserted in a class and never repeated again in your code.
5) Use Descriptive Names:
6) Function Arguments:
Common Monadic Forms (one argument function):
There are three reasons (three forms) to pass a single argument into a function.
Try to avoid any monadic functions that don’t follow these forms, for example, void includeSetupPageInto(StringBuffer pageText). Using an output argument instead of a return value for a transformation is confusing. If a function is going to transform its input argument, the transformation should appear as the return value.
Flag Arguments (Boolean arguments):
Flag arguments are ugly. It complicates the signature of the function. It makes the function does more than one thing, one thing if the flag is true and another if the flag is false. They are confusing and should be eliminated if possible.
Ex) Let's imagine we want to make a booking for a Hotel. There are two ways to do this, regular and premium. To use a flag argument here we would make a function declaration like this:
Then, when you (or anyone who read your code) see this function call book(customer, true), you can't remember what is the function do and what is the boolean argument means without going to read its implementation, but it's better if we separate it to two functions regularBook(customer) and premiumBook(customer), so, now it's easy to know what is the function do exactly from his call without need to go to its implementation.
Dyadic Functions (two arguments function):
A function with two arguments is harder to understand than a monadic function. But of course, there are times where two arguments are appropriate like Point p = new Point(0, 0); This is for sure because the points naturally take two arguments.
It will be better if you find a mechanism to convert the dyadic function to monadic (in some cases if possible). For example, writeField(outputStream, name) can convert to monadic by any one of these methods:
Triads Functions (three arguments function):
Function that takes three arguments is significantly harder to understand than dyadic. You should think very carefully before creating a triad.
Argument Objects:
When a function seems to need more than two or three arguments, then, it is likely that some of those arguments should be wrapped into a class of their own. For example,
can be wrapped to,
Recommended by LinkedIn
Argument Lists:
Sometimes we want to pass a variable number of arguments into a function. For example,
Then the declaration for this function will be as below,
Then, if the variable arguments are all treated identically, as they are in the example above, then they are equivalent to a single argument of type List (Object...).
So, all the previous rules apply. Functions that take variable arguments can be monads, dyads, or triads.
But it would be a mistake to give them more arguments than that.
7) Have No Side Effects:
Side effects mean that your function promises to do one thing, but it also does other hidden things. So, try to avoid these things because they leads to a temporal couplings (kind of coupling).
For example, the bellow function uses an algorithm to match a userName to a password. It returns true if they match. and false if anything goes wrong.
But, it also has a side effect is the call to Session.initialize(). The checkPassword function, by its name, says that it checks the password, but, the name don't says that it initializes the session.
Output Arguments
Arguments are often interpreted as inputs to a function, but sometimes interpreted as an outputs. For example, appendFooter(s); it's not obvious if that call takes 's' as an input and appends it to footer, or takes 's' as an output and appends footer to it. So, you have to go to the declaration to clarify the issue,
This clarifies the issue (you know now that 's' is an output), but only at the expense of checking the declaration of the function (this called double-take).
Output arguments should be avoided. If your function must change the state of something, have it change the state of its owning object. Like that report.appendFooter();
8) Command Query Separation:
Functions should either do something or answer something, but not both. For example:
This function sets the value of a named attribute and returns true if it is successful and false if no such attribute exists. This leads to odd statements like this:
This function confused the reader? Is it asking whether the “username” attribute was previously set to “unclebob”? Or is it asking whether the “username” attribute was successfully set to “unclebob”?
The solution is to separate the command from the query so that the ambiguity cannot occur.
9) Prefer Exceptions over Returning Error Codes:
When you return an error code, that leads to two problems,
Solution: if you use exceptions instead of returned error codes, then the error processing code can be separated from the happy path code. This make it more simple.
Extract Try/Catch Blocks:
Try/catch blocks are ugly and confuse the structure of the code and mix error processing with normal processing (happy path). So it is better to extract the bodies of the try and catch blocks out into functions of their own, like this:
Error Handling Is One Thing:
Functions should do one thing. Error handing is one thing. So, a function that handles errors shouldn't do another thing.
If the keyword try exists in a function, it should be the first word in the function and should be nothing after the catch/finally blocks.
Ex) The function delete(Page page) in the previous section.
The Error.java Dependency Magnet:
Returning error codes lead to that there is some class or enum in which all the error codes are defined.
Classes like this are a dependency magnet; many other classes must import and use them. Thus, when the Error enum changes, all those other classes need to be recompiled and redeployed.
10) Don’t Repeat Yourself:
Look up at Listing 3-1 carefully and you will notice that there is an algorithm that gets repeated four times,
This duplication was reduced by the include method in Listing 3-7. Read through that code again and notice how the readability of the whole module is enhanced by the reduction of that duplication.
11) Structure Programming:
Edsger Dijkstra’s rules of structured programming say that every function, and every block within a function, should have one entry and one exit. Following these rules means that there should only be one return statement in a function, no break or continue statements in a loop, and never, ever, any goto statements.
Master programmers think of systems as stories to be told rather than programs to be written.
Thanks for reading, waiting your feedback :)
Frontend Developer at Etiqueta Única
3yUncle Bob, in his book, says that the Listing 3-4 violates the SRP. How does this happens exactly?
Senior iOS Developer at Nabd نبض
4yNice ya bro go ahead.
Senior Android Developer @ Chain Reaction | Android Development
4yThanks