In the realm of software development, complexity can be a formidable adversary. While writing intricate, clever code may feel like a testament to one's skills, it can potentially be a bane when it comes to maintaining and debugging that code. This is where Kernighan’s Law comes into play, a principle named after Brian Kernighan that states, "Debugging is twice as hard as writing the code in the first place. Therefore, if you write the code as cleverly as possible, you are, by definition, not smart enough to debug it"1.
Brian Kernighan
Kernighan's Law is named for Brian Kernighan and derived from a quote from Kernighan and Plauger's book The Elements of Programming Style:
While hyperbolic, Kernighan's Law makes the argument that simple code is to be preferred over complex code, because debugging any issues that arise in complex code may be costly or even infeasible. Let's delve into this concept further by looking at three examples.
Examples
The Perils of Over-Engineering
Consider a software developer who decides to create a sorting algorithm from scratch for a simple task, such as sorting a list of names. They could easily use a built-in sorting function provided by most programming languages, but in an attempt to display their prowess, they construct an intricate and unique sorting mechanism. Later on, when a bug arises within this system, they find themselves in a quagmire, unable to comprehend the origins of the issue. Had they opted for the built-in function, they would have saved not only time but also the effort spent in debugging.
When Optimization Obscures the Code
In the quest for efficiency, a software engineer decides to optimize a piece of code that runs a critical operation. They successfully reduce the time complexity from O(n^2) to O(n log n), making the operation faster. However, this optimization introduces a layer of complexity that makes the code harder to understand and, in turn, more challenging to debug. When an unexpected issue surfaces, it takes them twice as long to identify the cause, which wouldn't have been the case with the original, simpler version of the code.
The Tangle of Multithreading
In an effort to improve performance, a team of developers decides to make their software application multithreaded. While this approach indeed speeds up the application, it introduces new bugs related to thread synchronization and deadlock, which are notoriously difficult to reproduce and fix. The cleverness employed in designing the multithreaded application becomes a roadblock in resolving these bugs.
These examples underline the key takeaway from Kernighan's Law: simplicity should always be preferred over complexity when writing code. This principle is crucial in the context of creating digital software products.
In software product development, maintainability and scalability are two pillars that ensure the product's longevity. Complex, clever code can hinder both these aspects. It can make onboarding new team members harder, slow down the feature development, and make debugging a time-consuming task. On the other hand, simple code improves readability, makes debugging easier, and fosters a more sustainable environment for growth and maintenance.
Conclusion
In conclusion, Kernighan’s Law serves as a reminder to software developers that, while it is tempting to demonstrate our skills through complex, clever code, the true craft lies in creating simple, effective, and maintainable solutions. In the long run, this approach leads to more robust and successful digital software products.