Your name is one of the most important parts of your identity. It is the same with code. Each module, class, function, and variable has a unique identity and fulfills a unique purpose identified by its name.
The precision and length of a name can correspond to the scope: if a name is seen often, it’s probably a highly reusable and generic entity. These entities can have shorter, more vague (reusable) and generic names. If a name is seen less often, it’s probably in a small scope and may be an implementation detail. These entities can have longer, more precise and narrow names. Using a generic name for a precise concept or a precise name for a generic concept forces readers to think about the concept in the wrong scope, and reduces readability.
The precision of naming takes away from the uniqueness of seeing — Pierre Bonnard [1]
Pierre Bonnard was an artist, and made the statement above in the context of art. Writing is also a form of art, as is writing software. I interpret this statement as claiming that the more precisely something is named, the less mysterious the thing is: the intention and functionality has been narrowed down. If the name is not precise enough, readers need to make assumptions, and may be wrong. If the name is too precise, readers may be overwhelmed with the level of detail, and the definition may be too brittle to survive a refactor.
I don’t mean to understate the difficulty of finding an appropriate name. However, if this is a consistent issue, it may be because the roles of the components in the system are not properly defined. It may also be due to a lack of understanding or application of design patterns. Design patterns have ubiquitous terminology. While one may unknowingly use design patterns in their code with their own custom terminology, a reader who is already familiar with design patterns would grasp the concepts quicker if existing terminology is reused.
When it comes to naming standards like capitalization, camelCase or snake_case and so on, it’s extremely important to consistently uphold those standards throughout the codebase. Naming standards can help a reader identify, at a glance, what the type or scope of something is. If the standards are not consistently applied and enforced, then readers will either not trust the standards, in which case they’re no longer effective, or worse, they will trust the standards and be misguided.
> My preferred naming conventions (click to expand)
Namespaces
- Namespaces must be lower case
- This allows differentiating free functions from static class member functions
- Allow common abbreviations but don’t enforce short namespaces
- Use one top level namespace with one nested layer such that cross-namespace interactions are reduced
- Exceptions to the one-nested-layer may be inline namespaces, such as for ABI versioning, or detail/implementation namespaces that aren’t meant to be seen by the users
Classes
- Classes must start with capital letters
- PascalCase
- Only the first letter in an acronym is capitalized, e.g. Rpc instead of RPC
Variables
- Lower case
- Forbid Hungarian Notation dependent on type
- Do not prefix/suffix pointer types with “p”, booleans with “b” and so on.
- The type of an object is less relevant than the purpose of the object. If the object survives a refactoring, it may have a different type but fulfill the same purpose.
- Must use Hungarian Notation dependent on scope
- The scope is relevant to the purpose of a variable
- Prefer adding an
m_
ands_
prefix for class member and static class member variables respectively. I don’t use non-const globals, but I might suggest ag_
prefix for those who do. - Prefer ALL_CAPS for const global variables or enumerations
- Discourage prefixing local variables or function parameters
- snake_case