Minimize Magic
Suppose I’m really excited for tomorrow. There’s a party going on or something I’m stoked on, and I’m jazzed up. Suppose further I want to express my excitement to you. There’s a few ways I could do that.
I could say “The day before the day after tomorrow is going to be a day which I will not want to forget to remember for a length of time that I could not say would ever not be indefinite!”. Or I could get a little fancy and say something like, “The simple thought of the jubilation of the forthcoming day fills me with such a sense of impending joy I find myself surreptitiously stirred.”
Or I could just say “I’m excited for tomorrow!”. You might use different terms depending on the company you keep but the points remains: I could use three different phrases to express the same thing.
Let’s look at some statistics for the phrases I could use:
1) “The day before the day after tomorrow is going to be a day which I will not want to forget to remember for a length of time that I could not say would ever not be indefinite!”
Words: 37
Double negatives: 3 [“not want to forget”, “could not say would ever not”, “not be indefinite”]
Elaborate words: 0
2) “The simple thought of the jubilation of the forthcoming hours fills me with such a sense of impending joy I find myself surreptitiously stirred.”
Words: 24
Double negatives: 0
Elaborate words: 3 [“jubilation”, “forthcoming”, “surreptitiously”]
3) “I’m excited for tomorrow!”
Words: 4
Double negatives: 0
Elaborate words: 0
1 and 2 both have some pretty fancy pomp and frill. They also require a lot more brain power to unpack. 3 conveys what it needs to without much fanfare. If you’re a fan of the English language and you simply appreciate the innate beauty and mystery of language you might prefer to use 1 or 2.
Some might even try to use phrases like 1 or 2 because they think that if they can convince the person they’re saying the phrase to that it requires a lot of brain power to unpack what they said that that means they’re smart. But if we take a step back and think about what we want to occur in this situation (me communicating something of meaning to you), then the most prudent choice of phrase would be 3.
It’s most likely you understand it, it’s easiest to say and it gets the job done. Code, in a lot of scenarios, is very similar. It’s natural to begin with the idea that machine language is very rigid and airy. That’s not an unfair thing to think because it certainly seems that way at first. It’s easy to get the impression that it’s this sort of mysterious set of arcane symbols that somehow makes the devices in front of you display the images and interpret the controls you input into them as you expect.
What’s more, modern pop culture in the US does computer science no favors by portraying it as this quasi-magical subject filled with nearly uninterpretable ideas and very strange, counterintuitive principles that requires hours of consideration to unwind. In a certain sense some of those things are true, but in a much more real sense code and computer science is not as esoteric or unapproachable as you might think.
I would be very surprised if code ever achieves the ubiquity that reading and writing natural language does as a skill, but nonetheless it is just as malleable and intuitive as natural language. All that is to say, just like natural language, there’s nothing about code being hard to understand that makes it good. It will require some persistence to understand in the best case, but if it’s filled with needless pomp and frill, like the phrases 1 and 2 above, it will be very hard to understand.
It can seem appealing to seek out the most complicated or counterintuitive areas of the system if you like to challenge yourself intellectually, but it’s important to remember that complicated and non-obvious are not the qualities of value when it comes to engineering. Conversely, simplicity and intuitiveness are very valuable qualities of a system. Moreover, eventually you’ll come to see it’s hard enough to make a system simple and intuitive: there is certainly no lack of intellectual challenge to go around.
It's really easy as a junior engineer to tread down the wrong path in this way and create a lot of needless complexity. All the senior engineers around you seem so smart and to have such an incredible depth of understanding of the system, but you don’t know what you don’t know. You certainly aren’t able to identify who’s just stuck around long enough and done a so-so job, or who phoned it in on Friday night to go have a drink with their pals and wrote a bunch of bloated, complicated software because they weren’t just weren’t interested enough to apply themselves.
Be wary of code that’s complicated or has many layers of indirection and non-obvious design properties. Often times that’s a red flag. Think about it - have you ever seen a variable in a source file that isn’t defined in the file or any of the imported modules? Maybe it’s a parameter automatically inserted by a unit test framework or some sort of code instrumentation tool?
Some magic just happens to plop the variable into place during execution, but it’s not obvious from looking at the code what the value will be or even what the parameters that will determine it are. Or maybe you built your own abstraction layers into a generic abstraction provided by a third-party library or you use some sort of interface binding to make code interoperable and some code which will end up in the same stack trace isn’t next to each other in the source, or maybe can’t even be linked together from looking at the source unless you knew about how the interface binding works?
This kind of stuff might sound fancy and exciting because it can pose an intellectual challenge to you as an engineer but really what you might find is that parts of a system with those qualities are only partially understandable by design. Someone had eyes larger than their brain and got themselves in too deep, or didn’t try very hard and just eked out a majorly wonky design or had to design it that way because there was an external constraint that made it unavoidable.
Whatever the case the end result is the system may literally be very hard or infeasible to fully model in your mind. That’s a huge problem because likely it means no one can trace down the bugs that might pop up and it won’t be clear what the impact of certain changes will be when you modify parts of the system that depend on the complicated one, or vice versa.
All that to say, if you start seeking out intellectual challenges for the sake of intellectual challenges will likely end up creating an unnecessarily convoluted system that is hard to maintain and modify.
The best code is simple and intuitive: it get’s the job done with the least pomp and frill, is the easiest to understand and is easy to change. The worst code is long-winded and filled with needless twists and turns that make it hard to figure out what’s going on and difficult to build on top of.