I’m still proud to have helped bring whitehouse.gov to WP VIP.
Josh Betz
Made with 🧀 in Madison
Made with 🧀 in Madison
I’m still proud to have helped bring whitehouse.gov to WP VIP.
When was the last time you opened a piece of code and thought, What is this even doing? Maybe you were debugging, adding a feature, or just trying to understand how it worked. Sometimes it’s your own code that trips you up. The problem isn’t that it’s broken—it’s that it’s hard to read.
Readable code matters more than we like to admit. It’s not just written once and forgotten; it’s read over and over again—by teammates, new hires, and sometimes by you, months or years later. In those moments, you will appreciate clear, simple code.
The temptation to future-proof everything is real. You imagine all the ways your code might need to change and try to prepare for them upfront. But here’s the thing: if you control the deployment pipeline, you don’t have to solve every problem now. Start simple. Simple code is easier to read, easier to understand, and, when the time comes, easier to adapt.
Developers spend far more time reading code than writing it. Whether you’re fixing a bug, adding a feature, or onboarding a new teammate, understanding the existing code is step one. If your code isn’t easy to read, it’s slowing everyone down.
Hard-to-read code creates friction. It makes debugging harder, onboarding slower, and feature development riskier. It also doesn’t age well. I’ve revisited code I wrote and found myself confused by my own decisions. If even the author can’t quickly understand the code, what chance does anyone else have?
Readable code isn’t a “nice-to-have.” It’s the foundation of maintainable software. The simpler and clearer your code is, the more effectively your team can work with it.
Readable code is simple code. It avoids unnecessary cleverness and prioritizes clarity over sophistication. A straightforward implementation may not win style points, but it will win over every developer who has to maintain it.
Consider this:
Simplicity is also a hedge against fragility. The fewer layers and abstractions your code has, the less likely it is to break or become difficult to debug. Clever code might feel satisfying to write, but simple code will serve your team better in the long run.
One of the most common mistakes developers make is over-engineering. You imagine all the ways your code might need to change someday and start adding abstractions to prepare. While the intention is good, the result is often needless complexity.
Abstractions aren’t bad by default. They can reduce duplication and encapsulate complexity when applied thoughtfully. But abstractions should solve real, present problems—not hypothetical ones. Over-abstraction creates layers of indirection that make your code harder to follow, not easier.
Before you add an abstraction, ask yourself:
If the answer to both isn’t a clear yes, skip it.
The beauty of writing programs you control is that you don’t need to solve every problem upfront. Libraries often have to support a wide range of use cases, requiring more extensibility and abstraction. But most programs are written to solve specific problems in specific contexts. They don’t need to work everywhere—they need to work here.
When you own the deployment pipeline, you have the luxury of starting simple. You don’t have to plan for every hypothetical scenario. If new needs arise, you can always refactor. It’s far easier to extend simple, readable code than to untangle an over-engineered solution.
For example, say you’re writing a program that interacts with a MySQL database. Instead of abstracting all database interactions just in case you switch to PostgreSQL later, write clear, direct queries. If you ever need to make a switch, your simple, readable code will make the transition easier—not harder.
How do you write code that’s easy to read and maintain? Here are some principles to guide you:
Readable code isn’t just about writing for yourself. It’s about making life easier for your teammates, future hires, and, yes, future you.
Code reviews are one of the most important tools for maintaining a healthy codebase. But too often, reviews focus on whether the code works rather than whether it makes sense. While correctness is crucial, it’s something automated tools like tests, linters, and type checkers should already be doing. A big value add of code reviews lies in enforcing clear, simple code.
A good code review asks questions like:
By prioritizing these questions, reviews become less about nitpicking details and more about ensuring the code is approachable for everyone who will work with it in the future. Reviews are your chance to ask, “Will the next person who touches this code understand it quickly?” If the answer is no, then it’s worth revisiting.
Readable, simple code is the foundation of sustainable software development. It makes your codebase easier to fix, easier to extend, and easier to trust. And when you keep things simple, you give your team the flexibility to adapt as new needs arise.
So the next time you sit down to write code, ask yourself:
If the answer is yes, you’re doing it right. Remember: it’s just code—it’s easy to change. Start simple, and let the future take care of itself.
Jeff Atwood, founder of Stack Overflow and Discourse:
The American Dream contains the path of hate, and the path of love. Throughout our history, one hand is always fighting the other. Which path are we choosing?
Our family pledges half our wealth toward an American Dream founded on love.
I didn’t know Warren Buffet was no longer giving his fortune to the Gates Foundation.
Last year he amended the pledge, giving all his wealth at death to a charitable trust run by his children, aged 71, 69, and 66, who do not make for natural charitable bedfellows.
Until recently, I assumed Tailscale was proxying most or all traffic through cloud servers. It turns out that it’s much cooler than that though. In most cases they can establish direct peer to peer connections. This article talks about how they traverse Network Address Translation and stateful firewalls.
Apple recommends adding some launch arguments and environment variables to your Xcode schemes to catch and debug Core Data problems.