Personal website of Martin Tournoij (“arp242”); writing about programming (CV) and various other things.

Working on GoatCounter and moreGitHub Sponsors.

Contact at martin@arp242.net or GitHub.

This page's author

Good comments should read well, be to the point, and don’t faff about with needless verbiage. This applies to any writing but doubly so for technical documentation and triply so for comments.

Quick example I found today:

// This block represents the list of errors that could be raised when using
// the webhook package.

Can be rephrased as:

// Possible errors when validating a webhook.

The default comment is adequate, but is a bit of a mouthful and can easily be rephrased to contain the exact same information. Why use more words if you don’t need to? It’s only distracting from the actual information you want to convey.

I don’t want to make the case to “use comment as little as possible” here – that’s a different discussion – just that comments should be to the point, and not faff about. The worst offenders is fluff like “Okay, let me explain what this function does [..]”, “This function will [..]”, “This setting will control [..]”, “Now that we are here [..]”. None of that adds any value.


It’s important to form sentences that read well. “if this condition is true, then do something” makes sense if you follow the code, but “do something if condition” is both shorter and usually reads more naturally. Comments sometimes seem like an unedited “stream of consciousness” that the programmer made while writing the code which were never edited to more standard language after being written.

Writing is hard and takes practice, but being attentive improves much of it and avoids the worst cases. I find that in my own comments the first attempt is often sub-optimal or downright bad, and requires some effort to get it right. Reading comments out loud helps a lot (this, again, is general writing advice).

It’s probably best to avoid slang, sayings, idioms, literary imagery, etc. as well. Your goal is to convey ideas, not to write an entertaining novel. I worked for an Irish company for three years, and was hella confused by all the craic my Irish coworkers wrote in the comments.

Spell check your comments. Yeah, misspelling “request” as “requst” isn’t a huge deal, but it does impede readability. Might as well get it right.

Code reviews should address comments just as the code, although I wouldn’t recommend being too pedantic about this. In the end, there is quit a lot of subjectivity involved in “good” writing.


While I don’t really want to enter the debate on when to comment in depth, some comments are truly redundant. Some of the silliest examples I have seen:

char buf[BUF_SIZE]; /* create an array of characters */

transaction.Commit() // commit the transaction

// Start the loop.
while ( have_posts() ) : the_post();

[..]

// End the loop.
endwhile;

Keep comments updated. If the code is updated, then also update the comment. Here’s some code I once encountered:

/**
 * Please be careful when modifying this function. This is really advanced code.
 * @return bool
 */
public static function initResources()
{
    return true;
}

While this example is obvious (even made it to The Daily WTF), most are more subtle.

Some examples

Some random examples I encountered.

Waffling:
// For card errors resulting from a card issuer decline, a short string
// indicating the card issuer’s reason for the decline if they provide
// one.
// Card issuer's reason for declining a card, if provided.
Waffling:
// DefaultTolerance indicates that signatures older than this will be rejected by ConstructEvent.
// Reject signatures older than this.
Awkward English:
# we want the modal to be draggable except if it's a fixed one
# The modal should be draggable unless it's fixed.
Needless language:
// If we're at this point, we should tell the client's browser what to send
// up in future
// Tell the client's browser what to send in future.
Needless language:
// Ok, we are good, so lets go ahead and start the merge now.
// Let's go ahead and remove duplicate threads
// Remove duplicate threads.
Needless language
// Before saving, check if this reply was from a forward to a 3rd party. If
// it was, then we need to save this as a note, not as a message
// Save as note if this reply was forwarded from a third party.
Confusing typo:
// good change it'll be cached
// Good chance it's cached.
Confusing language
if action.Notify.Subject == "" { // if subject is empty, it's mean that the user want a copy only
if action.Notify.Subject == "" { // The users wants only a copy if the subject is empty.
Confusing language; excessively verbose.
// This will check if the actual requst path ends in .json. If the router
// path that we defined ends in .json and the actual request URL does not
// end in .json, then we want to make this a not found error as it's not
// the correct URL.
// Make sure that "/foo/42" does *NOT* work when the route is
// "/foo/:id.json".
Confusing slang, I think? I don't have an alternative as I'm not sure what it's trying to say
// So we can't limit this in future cause it's horse
Overly verbose:
// RoundPlus will round  your float given the precision you specify: RoundPlus(7.258,2) will return 7.26
// RoundPlus will round the value to the given precision.
Confusing:
" get count of the motion. This should be done before all the normal
" expressions below as those reset this value(because they have zero
" count!). We abstract -1 because the index starts from 0 in motion.
" Get motion count; done here as some commands later on will reset it.
" -1 because the index starts from 0 in motion.
Verbose:
// Let's do a little bit of security before we try the delete.  Make sure that all of the tickets
// that are being passed in are valid tickets that the user can delete.
// Make sure the user is allowed to delete the tickets.
Verbose:
// Assert that the threads is bigger than 1 At this point there should
// be 2 threads in there - the original and this new one here
if numThreads < 1 {
// There should be two threads: the original and the new one.
if numThreads < 1 {
Confusing:
// findPhishingURLs finds all of the links in a given html string and returns a
// list of those that are known phising URLs.
// findPhishingURLs finds all known phising URLs in an HTML string.
Verbose, confusing:
// activityDeadline is the duration that an http request to any activity
// endpoint must complete within
// Maximum duration for activity endpoints.
Confusing, verbose:
# If we're pressing the down key and the next node is not
# content editable (i.e the signature box), then we want to fix
# the broken contenteditable behaviour by setting the cursor from
# where it is now to the end of the current node.
# The down key won't go to the end of line if the next node isn't
# contenteditable. This seems like a WebKit-specific bug we have to
# work around.