Links
Quotes
Chapter 1
When you have to add a feature to a program but the code is not structured in a convenient way, first refactor the program to make it easy to add the feature, then add the feature.
Let me stress that itβs these changes that drive the need to perform refactoring. If the code works and doesnβt ever need to change, itβs perfectly fine to leave it alone. It would be nice to improve it, but unless someone needs to understand it, it isnβt causing any real harm. Yet as soon as someone does need to understand how that code works, and struggles to follow it, then you have to do something about it.
Whenever I do refactoring, the first step is always the same. I need to ensure I have a solid set of tests for that section of code. The tests are essential because even though I will follow refactorings structured to avoid most of the opportunities for introducing bugs, Iβm still human and still make mistakes. The larger a program, the more likely it is that my changes will cause something to break inadvertently.
An important part of the tests is the way they report their results. They either go green, meaning that all the strings are identical to the reference strings, or red, showing a list of failuresβthe lines that turned out differently. The tests are thus self-checking. It is vital to make tests self-checking.
Before you start refactoring, make sure you have a solid suite of tests. These tests must be self-checking.
As I do the refactoring, Iβll lean on the tests. I think of them as a bug detector to protect me against my own mistakes. By writing what I want twice, in the code and in the test, I have to make the mistake consistently in both places to fool the detector. By double-checking my work, I reduce the chance of doing something wrong. Although it takes time to build the tests, I end up saving that time, with considerable interest, by spending less time debugging.
As I look at this chunk, I conclude that itβs calculating the charge for one performance. That conclusion is a piece of insight about the code. But as Ward Cunningham puts it, this understanding is in my headβa notoriously volatile form of storage. I need to persist it by moving it from my head back into the code itself. That way, should I come back to it later, the code will tell me what itβs doingβI donβt have to figure it out again.
The way to put that understanding into code is to turn that chunk of code into its own function, naming it after what it doesβsomething like amountFor(aPerformance). When I want to turn a chunk of code into a function like this, I have a procedure for doing it that minimizes my chances of getting it wrong. I wrote down this procedure and, to make it easy to reference, named it Extract Function (106).
IItβs an important habit to test after every refactoring, however simple. Mistakes are easy to makeβat least, I find them easy to make. Testing after each change means that when I make a mistake, I only have a small change to consider in order to spot the error, which makes it far easier to find and fix. This is the essence of the refactoring process: small changes and testing after each change. If I try to do too much, making a mistake will force me into a tricky debugging episode that can take a long time. Small changes, enabling a tight feedback loop, are the key to avoiding that mess.
Refactoring changes the programs in small steps, so if you make a mistake, it is easy to find where the bug is.
Itβs my coding standard to always call the return value from a function βresultβ. That way I always know its role.
Any fool can write code that a computer can understand. Good programmers write code that humans can understand.
Is this renaming worth the effort? Absolutely. Good code should clearly communicate what it is doing, and variable names are a key to clear code. Never be afraid to change names to improve clarity.
When Iβm breaking down a long function, I like to get rid of variables like play, because temporary variables create a lot of locally scoped names that complicate extractions. The refactoring I will use here is Replace Temp with Query (178).