Principles for Writing Good Code
Explore the necessity of writing good code and the key principles generally followed to create high-quality code.
The Necessity of Writing Good Code
If we focus solely on quickly writing code for immediate implementation, technical debt can accumulate to an unmanageable level, causing problems with future maintenance. Therefore, it goes without saying that writing readable and easily maintainable good code from the start is crucial when conducting a development project.
In the case of Problem Solving (PS) or Competitive Programming (CP), there’s an argument that since the code used to solve problems is typically not reused after the problem-solving session or competition ends, and especially in CP where there are time constraints, quick implementation might be more important than writing good code. To answer this question, it’s necessary to consider why you’re doing PS/CP and what direction you’re pursuing.
Personally, I think the points that can be learned through PS/CP are as follows:
- You can use and become familiar with various algorithms and data structures in the process of solving problems within given execution time and memory limits, which can help you develop a sense of which algorithms and data structures to use in specific situations when working on actual projects.
- After writing and submitting code, you can immediately receive objective feedback on correctness, execution time, and memory usage, allowing you to practice writing accurate code quickly and proficiently without missing any parts.
- You can compare your code with that of other experts and find areas for improvement.
- Compared to actual development projects, you repeatedly write smaller-scale code with similar functionality, so (especially when practicing PS alone) you can practice writing concise and good code while paying attention to details without being constrained by deadlines.
While some may enjoy PS/CP purely as a hobby, if you’re doing PS/CP indirectly to improve your programming skills, the last point of ‘practicing writing good code’ is just as significant as the first three. This is because writing good code doesn’t come naturally from the start but requires consistent practice through repetition. Moreover, complex and difficult-to-read code is hard to debug and not easy to write accurately in one go, so spending time on inefficient debugging often results in not implementing solutions quickly either. Although PS/CP is certainly quite different from professional work, completely disregarding writing good code and focusing solely on immediate implementation is, for the reasons mentioned above, putting the cart before the horse. Therefore, I personally think it’s better to write concise and efficient code even in PS/CP.
Comment added in December 2024:
Given the current trends, unless you’re majoring in computer science and planning to make development your profession, if you’re looking to use programming as a tool for numerical analysis or experimental data interpretation, it might be better to actively utilize AI tools like GitHub Copilot, Cursor, or Windsurf to save time and use that saved time to study other things. While there’s no reason to discourage those who enjoy PS/CP as a hobby, spending time and effort on PS/CP for the sake of practicing code writing seems to have a low cost-effectiveness ratio now. Even for development positions, I expect that coding tests as part of the hiring process will likely become significantly less important than before.
Principles for Writing Good Code
Whether it’s code written for competitions or for professional work, the conditions for good code are not significantly different. This article covers the main principles for writing generally good code. However, in PS/CP, there might be some compromises made relative to professional work for the sake of quick implementation, which will be mentioned separately in the article when applicable.
Writing Concise Code
“KISS (Keep It Simple, Stupid)”
- The shorter and more concise the code, the less likely it is to have typos or simple bugs, and the easier it is to debug.
- Write code that can be easily interpreted without separate comments as much as possible, and add detailed explanations with comments only when absolutely necessary. It’s preferable to maintain a concise code structure rather than relying on comments.
- If you do write comments, make them clear and concise.
- Limit the number of arguments passed to a function to three or fewer. If more arguments need to be passed together, group them into a single object.
Deeply nested conditional statements (double, triple depth) reduce readability, so increasing the depth of conditional statements should be avoided as much as possible. e.g., The code below using the Guard Clause idiom is more readable than the code above.
1 2 3 4 5 6 7 8 9 10
async def verify_token(email: str, token: str, purpose: str): user = await user_service.get_user_by_email(email) if user: token = await user_service.get_token(user) if token : if token.purpose == 'reset': return True return False
1 2 3 4 5 6 7 8 9 10 11 12
async def verify_token(email: str, token: str, purpose: str): user = await user_service.get_user_by_email(email) if not user: return False token = await user_service.get_token(user) if not token or token.purpose != 'reset': return False return True
However, in PS/CP, to further reduce code length for quick writing, there are cases where tricks like using macros in C/C++ are employed. While this can be useful occasionally in time-constrained competitions, it’s a method that only works in PS/CP, and generally, the use of macros in C++ should be avoided. e.g.,
1
#define FOR(i,n) for(int i=0; i<n; i++)
Code Modularization
“DRY (Don’t Repeat Yourself)”
- When the same code is used repeatedly, separate that part into functions or classes for reuse.
- Active reuse of code through modularization improves readability, and when code needs to be modified later, only the relevant function or class needs to be modified once, making maintenance easier.
- In principle, it’s ideal for one function to perform only one task and not do two or more things. However, the code written in PS/CP is usually a small-scale program performing simple functions, so there are limitations to reuse, and due to time constraints, it may be difficult to strictly follow principles as in professional work.
Utilizing Standard Libraries
“Don’t reinvent the wheel”
- While it’s useful to implement data structures like queues and stacks or sorting algorithms directly to understand the principles when studying algorithms or data structures, otherwise, it’s better to actively utilize standard libraries.
- Standard libraries have been used and verified countless times and are well-optimized, making them more efficient than implementing them again yourself.
- Using existing libraries eliminates the need to waste time unnecessarily implementing code with the same functionality, and it’s easier for other team members to understand your code during collaboration.
Using Consistent and Clear Naming Conventions
“Follow standard conventions”
- Use unambiguous variable and function names.
- Usually, each programming language has its own naming conventions, so learn the naming conventions used in the standard library of the language you’re using and consistently apply them when declaring classes, functions, variables, etc.
- Name variables, functions, and classes in a way that clearly shows what they do, and for boolean types, under what conditions they return True.
Normalize All Data for Storage
- Process all data in one consistent, normalized format.
- If the same data has two or more formats, subtle bugs that are difficult to catch can occur, such as slight differences in string representations or hash values.
- When storing and processing data such as time zones and strings, they should be converted to a single standard format like UTC or UTF-8 encoding as soon as they are received or calculated. It’s good to perform normalization from the start in the constructor of the class representing that data, or to perform normalization immediately in the function receiving the data.
Separate Code Logic and Data
Data unrelated to the code’s logic should not be directly inserted into conditional statements but separated into a separate table. e.g., It’s preferable to write code like the one below rather than the one above.
1 2 3 4 5 6
string getMonthName(int month){ if(month == 1) return "January"; if(month == 2) return "February"; ... if(month == 12) return "December"; }
1 2 3 4 5
const string monthName[] = {"January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December"}; string getMonthName(int month){ return monthName[month-1]; }