Comments can be used to convey what code does, what it should to, what it does not do, why it exists, when and how it should and shouldn’t be used, and more.
Let’s categorize them!
Isn’t this boring? Well, maybe, although Carl didn’t think so. And I see it as an important next step in our discussion about comments:
The different kinds of comments will be compared along the lines of their content, maintenance implications, locations and alternatives.
There are comments which narrate what the code does. Things like “loop through the list of customers” or “increase total by the new product’s price”.
For these comments to add any value at all, they must be absolutely up-to-date. Every diversion between code and comment will quickly lead to confusion and soon afterwards to them being ignored.
These are typically inline comments.
The only way to keep them halfway maintainable is to only ever let them reference code in the following few lines. Everything else falls apart quickly because there is no mechanism to find the far-away comments referencing the code one is about to change.
Narrations can be made almost superfluous by writing clean code. Careful naming, transparent design, employment of well-known patterns, etc. explain the code much better and also make it more maintainable.
It is widely agreed that these comments should be avoided. Only in rare cases, if arcane but unavoidable language mechanisms are used, might they be necessary.
Then there are contract-defining comments. They describe the central abstraction of a unit of code, how it interacts with its dependencies, and which pre- and postconditions hold.
They also talk about what the code does but unlike narrative comments they do so in a declarative style. Ideally, they can be read instead of code and tests.
Writing and maintaining comments that fulfill this promise requires determination, an eye for detail, and unambiguous language.
Successfully doing so will greatly increase the usability of the commented code. Unclear or outright wrong comments, on the other hand, or failing to keep contracts and code in sync will cause lots of confusion down the road.
Unlike narrations, contract comments only describe abstract behavior instead of implementation details and maintenance is hence less intense.
There is a clear imperative of locality. Contract comments preferably talk about single concepts and only mention other units of code (e.g. one method another) if absolutely unavoidable. If possible, this happens solely in the form of automatically updated links (e.g. @see in Javadoc). Otherwise the comments should avoid duplicating information and create a single source of truth.
Clean code and tests are often considered an alternative to contract comments.
The striking advantage of tests is that as long as the they are passing, the unit is guaranteed to display the tested behavior. The same is of course not true for comments where behavior and documentation can differ due to ambiguous wording and halfhearted maintenance.
A disadvantage of documenting solely in code and tests is that building a high level understanding from the bottom up requires context switches and takes time – potentially a lot. Much of the newly gained knowledge might soon be forgotten if the analyzed code is not worked with afterwards.
I consider the comparison of clean code and tests on one, and contract comments on the other side the most crucial aspect in this discussion. I will not further it here but repeat what I said before: Why not invest in both?
Comments can provide technical context and clarify what a unit code is there for. They can explain when it can be used and when it can not, which problem one might solve with it, and even give examples for how to do that.
Note that some of these information might also be part of the contract but it is important not to confound the two categories! Contract comments make a promise, context comments explain why it was made and what it’s good for.
Context comments will be of great value to anyone learning about that part of the code – be it to use it or to modify it.
In order to provide helpful context the author has to see things from the reader’s perspective. Something which can be difficult if routine-blindness strikes and all the abstractions and hard parts are so obvious.
Unlike contracts, context comments are not meant to replace reading and understanding the code itself – they only support that. So while they should be fairly up-to-date, some deviation from the code is tolerable.
It is essential to prevent the confusion of contract and context comments! It is hence best not to mix them or, if the language provides no other commonly used mechanism, at least separate them clearly.
In Java it is common to compose them as non-Javadoc blocks, usually at the beginning of the described unit of code. Since Java 8 the new tags @apiNote and @implNote can be used as well.
By their very nature comments of this kind might be less local and allude to other units of code. It is of course preferable to keep them local and find a meaningful place for them. This is not imperative, though, as they do not always have to be up-to-date.
The reader can usually find production code calling the unit she is investigating or have a look at its tests. Deriving a broader understanding from these call sites is similar to understanding a contract from its tests: it is always up-to-date but can be a tedious bottom-up approach. Additionally this might mix intended and problematic use cases without further distinction or explanation.
Demos are a great way to describe what a piece of code is there for. They are rare, though, and must be maintained as well.
Code is often written according to some specification or situation that demands a non-obvious design. Maybe a cleaner approach could’ve been taken were it not for some very specific detail that could otherwise not be incorporated. Documentation explaining such circumstances provides valuable information to understand why the code exists.
One form to document historical context are code comments.
They should always be taken with a grain of salt and be written in a form which prevents confusion with other comments. Ideally historical context comments include some phrase like “at the time of writing” to emphasize their transient nature. It is easier to identify them as outdated if they clearly state their assumptions.
I see little reason to invest time in updating them. It seems better to let them stand even if slightly aged and simply delete them when the circumstance they describe changed too much for them to be useful.
Maybe even more often than comments providing technical context will these describe or reference concepts that do not have a natural singular location. This is tolerable if it is agreed upon that they are not updated.
In the form of commit messages these information can also be added to source control. Doing so has the advantage that commit messages are always “up-to-date” as the changeset itself is immutable. A message can also cover a set of changes across different files, which often comes in handy. At the same time, this can be a disadvantage because very local information must be searched searched across several larger messages. Finally, looking up commit messages adds a level of indirection from code to documentation.
Another source for historical context are the issues the commits are associated with. They can contain lots of valuable information that are hard to present in comments or commit messages, like diagrams, discussions with clients, links to other sources like Wikis, and more. But issues are even further removed from the code and often cover more high-level ground.
We divided comments into these four categories:
|very low||point out
I hope this taxonomy helps teams to decide whether and how they want to use comments. We can see that different kinds have different properties and should hence be discussed separately.