These are mostly notes from the classic "Clean Code" by Uncle Bob.
Use meaningful variable/function/class names.
- Don't use magic constants. Always make a variable. Easier to refactor later!
- Use descriptive names:
userPhone
,customerId
.. Don't make me guess! - Don't prefix interfaces with
I
, e.g.IMyService
.. Why should a user care that it is an interface???
That's the whole point of interface, to abstract away the details - You can use single-letter name but only in a narrow context, lambda or a for loop..
- Don't encode type in the name, e.g.
userString
... We have compilers now! - Be consistent! If you used
UserRepo::findByName()
, don't makeUserRepo::getByEmail()
.. it's confusing and rude! :D - Avoid redundant information! If you have
class UserRepo
, why would its method be calledfindUsers()
and not justfind()
?
We already know it is a user repo!!
- Do one thing!
- Smaller - the better!
- Less parameters - the better! (make a class for grouping them! bonus: easier refactoring)
- Avoid side effects
- Keep the same level of Abstraction in a function!
If you're talking about cars, you shouldn't suddenly jump to atoms!
Same with controller->regex-parser e.g. - If you side-effect, don't return a value; If you query, don't side-effect! (command-query separation)
Although Java violates this one regularly, e.g.int remove(Object o)
- Minimize use of switch statements. Preferably only in factory methods!
That's because if you use them extensively, you have to update switches all over the place when you add another case!! Compared to polymorphic class/interface where you don't have to! :) - Prefer exceptions to error codes! Happy path is much easier to read/understand.
Codes poison your calling methods..if(myFun() == OK)..
all over the place. Unless you use a Monad or some wrapper with utility methods, but then you propagate that special type.. (Optional<T>
for example) :D
Use them only when you really have to!
- Clarifications to unusual/specific/performace-related code is ok
- Warnings are ok..
- TODOs are ok (usually link the issue or something..)
- Javadocs are great!
- Don't comment out code, delete it!
- Use company/project-wide settings and obey them! Commit it to git, or put into Jira somewhere.
- Use automatic formatter if possible, a Git hook or something.
- One blank line between functions/classes
- Declare variable near to its usage!
- Line length should be 80+, I prefer 120
- Put related functions near each other. I prefer to put private ones on the bottom
Procedural code makes it hard to add new data structures because all the functions must
change.
OO code makes it hard to add new functions because all the classes must change.
(Wadler's problem? typeclasses, visitors)
Also https://en.wikipedia.org/wiki/Law_of_Demeter (not that important IMHO)
- Use exceptions! This is not C, you have a powerful mechanism at your hands.
- Use unchecked exceptions! Checked ones just clutter your method signatures and don't add much value to your code...
- Log the errors!!!
- Don't return
null
!!! Use null-object pattern, empty list/map,Optional<T>
or throw exception.. - Don't pass
null
! Try to find another way, overload function etc.
Write clean and useful tests! Fast, independent, repeatable!
Use IOC principle! Instantiate/wire all needed classes in the main
method!
Easier to understand, refactor and test. Use factories/DI if needed.