This is a topic much discussed in the industry and the position I take is a common one: Code first for readability and support, refactor for performance as necessary. There are notable exceptions to the wisdom of taking a readability-first approach. In general you will know if your code is one of these exceptions. It could be graphics-intensive operations, operations widely used and distributed in libraries, and similar situations where you know critical paths through the code a priori. For the 90% case, where squeaking out every nanosecond of performance is not critical I like to use the following steps:
- Write code to be as readable and supportable as possible
- Validate that the performance is acceptable
- If performance is not acceptable, profile to find the hot spots
- Identify techniques for improving performance of these hot spots
- Using the Strategy Pattern
- Extract each module to be refactored into a separate component and introduce a simple factory to serve up the existing implementation
- Introduce a higher performing implementation and update the factory to use this new strategy
- Add comments as necessary to the new strategy implementation
I am of the school of thought that a lot of comments in code is a design smell that the code wasn't as well thought out and decomposed as it should be. I think that most code should be written to be self-documenting with appropriately named methods with small scope. There are certain exceptions to this. Writing for performance is a common way of adding unintentional complexity. Often this added complexity is unavoidable. We should add sufficient comments to explain the code and the reasoning behind the extra complexity so that someone further down the road does not reverse our decisions (perhaps for readability) without understanding the implications of that reversal.
Next week I will push into a use-case where I opted for readability over performance in conflict with published performance recommendations from the platform vendor.