Clean Code | Chapter(3) | Functions

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:

  • Should be at maximum 20 lines long.
  • The blocks within if, else, while, and so on should be one line long. Probably that line should be a function call.
  • The nested structures (blocks) should be one or two at maximum.
  • See Listing 3-1 how it's refactored to be Listing 3-2, and then refactored again to be shorten to Listing 3-3.

Listing 3-1 | HtmlUtil.java

No alt text provided for this image

Listing 3-2 | HtmlUtil.java (refactored)

No alt text provided for this image

Listing 3-3 | HtmlUtil.java (re-refactored)

No alt text provided for this image


2) Do One Thing:

  • Functions should do ONE thing. They should do it WELL. They should do it ONLY.
  • To know the function do one thing or not, try to describe it as an only one TO paragraph, let's take Listing 3-3 as an example, we converted it to following TO paragraph:

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.

  • Look above at Listing 3-1 is doing more than one thing. It’s creating buffers, fetching pages, searching for inherited pages, rendering paths, and so on.. It contains many different levels of abstraction (many TO paragraphs).
  • Even Listing 3-2 has two levels of abstraction (two TO paragraphs).
  • Functions that do one thing cannot be reasonably divided into sections.


3) One Level of Abstraction per Function:

  • The Stepdown Rule (Reading Code from Top to Bottom):

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

No alt text provided for this image


No alt text provided for this image


4) Switch Statements:

No alt text provided for this image

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,

No alt text provided for this image

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.

No alt text provided for this image

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:

  • Don’t be afraid to make a name long. A long descriptive name is better than a short unclear name. A long descriptive name is better than a long descriptive comment.
  • Don’t be afraid to spend time choosing a name.
  • Choosing descriptive names will clarify the design of the module in your mind and help you to improve it.
  • Be consistent in your names. Use the same phrases, nouns, and verbs in the function names that you choose for your modules.


6) Function Arguments:

  • The ideal number of arguments for a function is zero.
  • Three arguments should be avoided as possible.
  • More than three requires a special justification.. and shouldn’t be used anyway.

Common Monadic Forms (one argument function):

There are three reasons (three forms) to pass a single argument into a function.

  1. Function that asking a question about that argument: Ex) boolean fileExists(“MyFile”).
  2. Function that transforming the argument into something else and returning it: Ex) InputStream fileOpen(“MyFile”) transforms a file name String into an InputStream return value.
  3. Function is an event: in this form there is an input argument but no output argument, and the function use the argument to alter the state of the system Ex) void passwordAttemptFailedNtimes(int attempts)

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:

No alt text provided for this image

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.

No alt text provided for this image

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:

  1. Make the writeField a member function of OutputStream class so that you can say outputStream.writeField(name)
  2. Or make the outputStream a member variable of the current class so that you don’t have to pass it, like this writeField(name).

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,

No alt text provided for this image

can be wrapped to,

No alt text provided for this image

Argument Lists:

Sometimes we want to pass a variable number of arguments into a function. For example,

No alt text provided for this image

Then the declaration for this function will be as below,

No alt text provided for this image

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.

No alt text provided for this image

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.

No alt text provided for this image

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,

No alt text provided for this image

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:

No alt text provided for this image

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:

No alt text provided for this image

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.

No alt text provided for this image


9) Prefer Exceptions over Returning Error Codes:

When you return an error code, that leads to two problems,

  • Deeply nested structures.
  • The caller must deal with the error immediately.

No alt text provided for this image

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.

No alt text provided for this image

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:

No alt text provided for this image

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.

No alt text provided for this image

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,

No alt text provided for this image

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 :)
Jorge Camargo

Frontend Developer at Etiqueta Única

3y

Uncle Bob, in his book, says that the Listing 3-4 violates the SRP. How does this happens exactly?

Like
Reply
Mohamed Nabih

Senior iOS Developer at Nabd نبض

4y

Nice ya bro go ahead.

Mai Abu ajamieh

Senior Android Developer @ Chain Reaction | Android Development

4y

Thanks

To view or add a comment, sign in

More articles by Mahmoud Ibrahim

  • Clean Code | Chapter(6) | Objects and Data Structures

    Clean Code | Chapter(6) | Objects and Data Structures

    Data Abstraction: Abstraction means hiding implementation. Hiding implementation isn't means making the variables…

    7 Comments
  • Clean Code | Chapter(5) | Formatting

    Clean Code | Chapter(5) | Formatting

    Code formatting is important. Code formatting is about communication, and communication is the professional developer’s…

    1 Comment
  • Clean Code | Chapter(4) | Comments

    Clean Code | Chapter(4) | Comments

    This chapter talks about some rules should be followed when writing the comments, and some cases which writing a…

    2 Comments
  • Clean Code | Chapter(2) | Meaningful Names

    Clean Code | Chapter(2) | Meaningful Names

    The following 15 rules for creating a good names for any thing in your software: variables, functions, arguments…

    18 Comments

Insights from the community

Others also viewed

Explore topics