Get to Intent Revealing (Article 6)
8 minute read
Every improvement in our name until now has focused on what our class/method/variable does. We made it honest and then we made it do exactly what we want. The name is both correct and clear.
But it is awkward.
We want to use this name to understand calling methods. It doesn’t yet serve that purpose. The name describes the thing. It doesn’t tell us why we care.
That makes us stumble when reading the caller. We want the names to tell us the purpose of each subcomponent. But right now the name tells us what it does—implementation when we want behavior/intent.
Every time we read a caller, we have to consider whether that implementation will give us our desired intent. That takes mental energy we could better spend elsewhere.
Part 1: Look at the calling methods / use sites
We’ve gotten as far as we can looking at our target to gain insights. Now we need to include the insights from its context.
Read through all the places where this thing is used. Understand it in the flow of other classes and method calls. What happens before, after, or with it?
Most likely the other names around this one will suck too. You need to improve them before you gain any insight. Sometimes insight will happen if the other parts are Honest, but you probably need to get to at least Completely Honest.
When you see all the context items at the Completely Honest level, you may realize that your target actually doesn’t Do the Right Thing. This is rare but it does happen. Fix it.
Part 2: Look for the code’s purpose
As you start to examine the use contexts you will start to see a flow. The target is one step in a larger orchestration. It has a purpose. That purpose is usually at a higher level of abstraction.
Rename the thing to state this purpose. Some common purposes are:
Methods:
- What it accomplishes (post-condition), regardless of starting state or process.
- Transformation it makes, regardless of beginning or ending states.
- Business process it replaces.
- Core responsibility the method takes on.
- What the method promises to allow callers to safely ignore.
Classes:
- The common responsibility of its methods.
- What the class is in the real world (Whole Value).
- The responsibility that calling context is delegating to the class and assuming the class will take care of. What the caller wants to ignore.
Variables:
- How this instance differs from other instances of the same type.
- The purpose the context has for this instance. How it is used.
Part 3: Write down the purpose by renaming the target
In our running example, storeFlightToDatabaseAndShowOnScreenIfVisible(flight)
becomes beginTrackingFlight(flight)
. The first name tells us what our method does. The second tells us why we care.
We don’t need to know what tracking a flight entails. We just need to know that this method will take care of initiating that process for us.
We would expect to see a stopTrackingFlight(flight)
nearby. And perhaps some query methods to find information about flights that are being tracked.
Play it again
Now repeat this step for another calling context. Your insight may be the same or your insightful name might be jarring in the other context. Often multiple callers are using the same code because of what it happens to do, not because of its purpose.
When this happens, create two methods that share an implementation. To the outside world, they are different things because they express different intent. It just happens that to a computer they do the same thing…for now.
Refactoring methods to share implementation
Do the following:
- Extract method the entire body of the method.
- Rename the extracted (inner) method to the name you had that Does the Right Thing (which you can see in source history).
- Duplicate the outer method and edit its name to unique nonsense (do not refactor; you don’t want to update callers).
- Rename the unique nonsense to the new purpose.
- Edit the call site to use the method with the new purpose.
The reason we hand-edit to nonsense and then rename is so that our tool can catch errors. Hand-editing code is dangerous and something I tend to avoid.
I could duplicate a name or create aliasing problems, especially in the presence of inheritance or global functions with imported namespaces. But these are all problems that the rename refactoring checks for. By hand-editing to unique nonsense, I reduce the chance of making a mistake. And then renaming to the final name ensures that I don’t introduce one at that point.
Refactoring classes to share implementation
For a class, the recipe depends on how your IDE does Split Class.
The easy way: use tools
If Split Class supports delegation:
- Split Class and select everything. Choose the option to have all callers use the old type and create delegating members.
- Rename the inner type to the name you had after Doing the Right Thing.
- Copy the file for the outer type (assuming class-per-file).
- Hand-change the class name & constructors in the copied file to something guaranteed unique. Nonsense is good here.
- Rename the copied class to the new purpose.
- Update the construction site to instantiate the new class.
- Follow the compile errors, applying the auto-fix each time to change the type of the variable.
The hard way: use a shim
If Split Class doesn’t support delegation:
- Create a new class with a unique nonsense name, a single private field of the old type and constructors that match the old type. Use auto-generate members to create delegating members.
- Rename the old class back to its Does The Right Thing name to make sure that doesn’t cause conflicts.
- Undo the rename refactoring (it was just a test).
- Hand-edit the name of the outer class to the name the inner class has (the Intent name for the other context).
- Hand-edit the inner class to the Does the Right Thing name.
- Update the type and construction of the field in the outer class to use the correct type.
- At this point, all call sites will now be using the outer class via the purpose-based name.
- Pick up from step 3 in the recipe that assumed Split Class that supports delegation.
This recipe uses a common mini-recipe: introduce a new name for an extracted thing but leave all users using the outer thing.
If we can’t extract the inner thing, then we can get the same result by creating a new outer thing as a shim and then hand-editing the names (not updating call site) to slip it into place.
It also uses a common refactoring test strategy. The most useful part of refactorings is not that they change your code, it is that they tell you what code changes are safe. Sometimes a refactoring won’t do exactly what you want, but you can still use its analysis to verify that the thing you want to do is safe.
We do and undo a rename refactoring to test whether our new names will create any conflicts, even though we execute the name changes by hand so that we can put the shim into place.
Warning: you can get nonsense
Naming by Intent is the easiest one to screw up. If you screw it up you will get a nonsense name. You won’t notice; the name will make sense to you right now because you have the full context. But it will be nonsense the next time you try to use it.
The problem is that this step is intentionally information-losing. We are replacing some of the information about what our thing does with information about why you care to use the thing. We are asking potential users to trust us. And we have to earn that trust. We earn it by using consistently clear names and code that Does the Right Things.
Ways to get nonsense
An extremely popular way to get nonsense is to name a thing by when it is used, rather than why it is used. This gets us right back to calling something preLoad()
. One variation on this is to name a method by its initial condition. Ignore initial conditions in method names; they always end in broken abstractions.
Another popular route to nonsense is to create a new concept that is never used anywhere else. One-off abstractions are noise. Create your abstraction by looking at many named entities and use it uniformly.
But perhaps the most popular route to nonsense is to use a CS-y term. The domain of software engineering does not belong in your names unless you are writing a programming language. Even then, domain-specific languages are easier to read.
To avoid nonsense be specific about the context. Clearly state why someone would want to do the things that this class/method is doing. Keep any parts of the Complete and Honest name that make it more obvious when you want to use this thing. Drop the rest.
Adopt Now
Help your team adopt Naming as a Process now!
Read More
Next up: see how our intention-revealing names make the right Domain Abstractions obvious.
Or read the whole Naming as a Process blog series.