Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/webpro/programming-principles

Categorized overview of programming principles & design patterns
https://github.com/webpro/programming-principles

architectural-patterns design-patterns programming-principles

Last synced: about 1 month ago
JSON representation

Categorized overview of programming principles & design patterns

Awesome Lists containing this project

README

        

# Programming Principles

Every programmer benefits from understanding programming principles and design
patterns. This overview is a reference for myself, and maybe it is of help to
you during design, discussion, or review. The list is incomplete, and sometimes
trade-offs need to be made between conflicting principles. However, higher
ranked principles usually beat lower ranked ones.

The list was inspired by
[The Principles of Good Programming](https://www.artima.com/weblogs/viewpost.jsp?thread=331531)
(down?
[Google's cache](https://webcache.googleusercontent.com/search?q=cache:KU51T8hZ-0kJ:https://www.artima.com/weblogs/viewpost.jsp%3Fthread%3D331531+&cd=1&hl=en&ct=clnk&gl=nl&client=pub-3911176865765226)).
I've added a bit more reasoning, details, and links to further resources.
[Let me know](https://github.com/webpro/programming-principles/issues) if you
have any feedback or suggestions for improvement!

## Contents

### Generic

- [KISS](#kiss)
- [YAGNI](#yagni)
- [Do The Simplest Thing That Could Possibly Work](#do-the-simplest-thing-that-could-possibly-work)
- [Separation of Concerns](#separation-of-concerns)
- [Code For The Maintainer](#code-for-the-maintainer)
- [Avoid Premature Optimization](#avoid-premature-optimization)
- [Optimize for Deletion](#optimize-for-deletion)
- [Keep things DRY](#keep-things-dry)
- [Boy Scout Rule](#boy-scout-rule)

### Relationships between modules, classes, components, entities

- [Connascence](#connascence)
- [Minimise Coupling](#minimise-coupling)
- [Law of Demeter](#law-of-demeter)
- [Composition Over Inheritance](#composition-over-inheritance)
- [Orthogonality](#orthogonality)
- [Robustness Principle](#robustness-principle)
- [Inversion of Control](#inversion-of-control)

### Modules, classes, components, entities

- [Maximise Cohesion](#maximise-cohesion)
- [Liskov Substitution Principle](#liskov-substitution-principle)
- [Open/Closed Principle](#openclosed-principle)
- [Single Responsibility Principle](#single-responsibility-principle)
- [Hide Implementation Details](#hide-implementation-details)
- [Curly's Law](#curlys-law)
- [Encapsulate What Changes](#encapsulate-what-changes)
- [Interface Segregation Principle](#interface-segregation-principle)
- [Command Query Separation](#command-query-separation)
- [Dependency Inversion Principle](#dependency-inversion-principle)
- [SOLID](#solid)

### Test

- [FIRST principles of testing](#first-principles-of-testing)
- [Arrange, Act, Assert](#arrange-act-assert)

## KISS

**Keep It Simple, Stupid**: most systems work and are understood better if they
are kept simple rather than made complex.

Why

- Less code takes less time to write, has less bugs, and is easier to modify.
- Simplicity is the ultimate sophistication.
- It seems that perfection is reached not when there is nothing left to add, but
when there is nothing left to take away.

Resources

- [KISS principle (wikipedia.org)](https://en.wikipedia.org/wiki/KISS_principle)
- [Keep It Simple Stupid (KISS) (principles-wiki.net)](http://principles-wiki.net/principles:keep_it_simple_stupid)
- [Cognitive Load is what matters (github.com)](https://github.com/zakirullin/cognitive-load)
- [The Limits of Human Cognitive Capacities in Programming and Their Impact on Code Readability (florian-kraemer.net)](https://florian-kraemer.net/software-architecture/2024/07/25/The-Limits-of-Human-Cognitive-Capacities-in-Programming.html)
- [Simplicity is An Advantage but Sadly Complexity Sells Better (eugeneyan.com)](https://eugeneyan.com/writing/simplicity/)

## YAGNI

**You Aren't Gonna Need It**: don't implement something until it is necessary.

Why

- Any work that's only used for a feature that's needed tomorrow, means losing
effort from features that need to be done for the current iteration.
- It leads to code bloat; the software becomes larger and more complicated.

How

- Always implement things when you actually need them, never when you just
foresee that you need them.

Resources

- [You Arent Gonna Need It (wiki.c2.com)](http://c2.com/xp/YouArentGonnaNeedIt.html)
- [You’re NOT gonna need it! (ronjeffries.com)](https://ronjeffries.com/xprog/articles/practices/pracnotneed/)
- [You aren't gonna need it (wikipedia.org)](https://en.wikipedia.org/wiki/You_aren%27t_gonna_need_it)

## Do The Simplest Thing That Could Possibly Work

Why

- Real progress against the real problem is maximized if we just work on what
the problem really is.

How

- Ask yourself: "What is the simplest thing that could possibly work?"

Resources

- [Do The Simplest Thing That Could Possibly Work (wiki.c2.com)](http://c2.com/xp/DoTheSimplestThingThatCouldPossiblyWork.html)

## Separation of Concerns

Separation of concerns is a design principle for separating a computer program
into distinct sections, such that each section addresses a separate concern. For
example, the business logic and the user interface of an application are
separate concerns. Changing the user interface should not require changes to the
business logic and vice versa.

Quoting [Edsger W. Dijkstra](https://en.wikipedia.org/wiki/Edsger_W._Dijkstra)
(1974):

> It is what I sometimes have called "the separation of concerns", which, even
> if not perfectly possible, is yet the only available technique for effective
> ordering of one's thoughts, that I know of. This is what I mean by "focusing
> one's attention upon some aspect": it does not mean ignoring the other
> aspects, it is just doing justice to the fact that from this aspect's point of
> view, the other is irrelevant.

Why

- Simplify development and maintenance of software applications.
- When concerns are well-separated, individual sections can be reused, as well
as developed and updated independently.

How

- Break program functionality into separate modules that overlap as little as
possible.

Resources

- [Separation of Concerns (wikipedia.org)](https://en.wikipedia.org/wiki/Separation_of_concerns)

## Code For The Maintainer

Why

- Maintenance is by far the most expensive phase of any project.

How

- _Be_ the maintainer.
- Always code as if the person who ends up maintaining your code is a violent
psychopath who knows where you live.
- Always code and comment in such a way that if someone a few notches junior
picks up the code, they will take pleasure in reading and learning from it.
- [Don't make me think](http://www.sensible.com/dmmt.html).
- Use the
[Principle of Least Astonishment](https://en.wikipedia.org/wiki/Principle_of_least_astonishment).

Resources

- [Code For The Maintainer (wiki.c2.com)](http://wiki.c2.com/?CodeForTheMaintainer)
- [The Noble Art of Maintenance Programming (blog.codinghorror.com)](https://blog.codinghorror.com/the-noble-art-of-maintenance-programming/)

## Avoid Premature Optimization

Quoting [Donald Knuth](https://en.wikiquote.org/wiki/Donald_Knuth):

> Programmers waste enormous amounts of time thinking about, or worrying about,
> the speed of non-critical parts of their programs, and these attempts at
> efficiency actually have a strong negative impact when debugging and
> maintenance are considered. We should forget about small efficiencies, say
> about 97% of the time: premature optimization is the root of all evil. Yet we
> should not pass up our opportunities in that critical 3%.

Understanding what is and isn’t "premature" is critical.

Why

- It is unknown upfront where the bottlenecks will be.
- After optimization, it might be harder to read and thus maintain.

How

- [Make It Work Make It Right Make It Fast (wiki.c2.com)](http://wiki.c2.com/?MakeItWorkMakeItRightMakeItFast)
- Don't optimize until you need to, and only after profiling you discover a
bottleneck to optimise.

Resources

- [Program optimization (wikipedia.org)](https://en.wikipedia.org/wiki/Program_optimization)
- [Premature Optimization (wiki.c2.com)](http://wiki.c2.com/?PrematureOptimization)

## Optimize for Deletion

Why

Code should be optimized for change. Code that's easy to delete is easy to
replace or change.

How

Don't try to predict future changes today, instead focus on deletable and
rewriteable code.

Principles such as [separation of concerns](#separation-of-concerns),
[low coupling](#minimise-coupling) and the
[Single Responsibility Principle](#single-responsibility-principle) result in
increased deletability. In general, high modularity helps to increase
deletability of individual parts.

Resources

- [The art of destroying software (youtube.com](https://www.youtube.com/watch?v=Ed94CfxgsCA)
- [Optimize for Deletion (work.stevegrossi.com)](https://work.stevegrossi.com/2016/11/04/optimize-for-deletion/)
- [Deletability (kellysutton.com)](https://kellysutton.com/2017/05/29/deletability.html)
- [Write code that is easy to delete, not easy to extend.](https://programmingisterrible.com/post/139222674273/write-code-that-is-easy-to-delete-not-easy-to)

## Keep things DRY

**Don't Repeat Yourself**: Every piece of knowledge must have a single,
unambiguous, authoritative representation within a system.

Each significant piece of functionality in a program should be implemented in
just one place in the source code. Where similar functions are carried out by
distinct pieces of code, it is generally beneficial to combine them into one by
abstracting out the varying parts.

Why

- Duplication (inadvertent or purposeful duplication) can lead to maintenance
nightmares, poor factoring, and logical contradictions.
- A modification of any single element of a system does not require a change in
other logically unrelated elements.
- Additionally, elements that are logically related all change predictably and
uniformly, and are thus kept in sync.

How

- Put business rules, long expressions, if statements, math formulas, metadata,
etc. in only one place.
- Identify the single, definitive source of every piece of knowledge used in
your system, and then use that source to generate applicable instances of that
knowledge (code, documentation, tests, etc).
- Apply the
[Rule of three]().

Resources

- [Dont Repeat Yourself (wiki.c2.com)](http://wiki.c2.com/?DontRepeatYourself)
- [Don't repeat yourself (wikipedia.org)](https://en.wikipedia.org/wiki/Don't_repeat_yourself)
- [DRY Principle: Its Benefit and Cost with Examples (thevaluable.dev)](https://thevaluable.dev/dry-principle-cost-benefit-example/)

Related

- [Abstraction principle (wikipedia.org)]()
- [Once And Only Once (wiki.c2.com)](http://wiki.c2.com/?OnceAndOnlyOnce) is a
subset of DRY (also referred to as the goal of refactoring).
- [Single Source of Truth (wikipedia.org)](https://en.wikipedia.org/wiki/Single_Source_of_Truth)
- A violation of DRY is [WET](http://thedailywtf.com/articles/The-WET-Cart)
(Write Everything Twice)
- [Be careful with the code metric "duplicated lines" (rachelcarmena.github.io)](https://rachelcarmena.github.io/2018/02/27/duplication-you-are-welcome.html)

## Boy Scout Rule

The Boy Scouts of America have a simple rule that we can apply to our
profession: "Leave the campground cleaner than you found it". The boy scout rule
states that we should always leave the code cleaner than we found it.

Why

- When making changes to an existing codebase the code quality tends to degrade,
accumulating technical debt. Following the boy scout rule, we should mind the
quality with each commit. Technical debt is resisted by continuous
refactoring, no matter how small.

How

- With each commit make sure it does not degrade the codebase quality.
- Any time someone sees some code that isn't as clear as it should be, they
should take the opportunity to fix it right there and then.

Resources

- [Opportunistic Refactoring (martinfowler.com)](https://martinfowler.com/bliki/OpportunisticRefactoring.html)

## Connascence

Connascence is a software quality metric that describes different levels and
dimensions of coupling. Two components are connascent if a change in one would
require a change in the other as well. Lower connascence means higher code
quality.

Why

- Reducing connascence will reduce the cost of change for a software system.
- Arguably one of the most important benefits of connascence is that it gives
developers a vocabulary to talk about different types of coupling.

How

- Each instance of connascence in a codebase is considered on 3 separate axes:
strength, degree and locality.
- It provides a system for comparing different types of dependency.

Resources

- [Connascence (connascence.io)](https://connascence.io)
- [Connascence (wikipedia.org)](https://en.wikipedia.org/wiki/Connascence)

## Minimise Coupling

Coupling between modules or components is their degree of mutual
interdependence; lower coupling is better. In other words, coupling is the
probability that code unit "B" will "break" after an unknown change to code unit
"A".

Why

- A change in one module usually forces a ripple effect of changes in other
modules.
- Assembly of modules might require more effort and/or time due to the increased
inter-module dependency.
- A particular module might be harder to reuse and/or test because dependent
modules must be included.
- Developers might be afraid to change code because they aren't sure what might
be affected.

How

- Eliminate, minimise, and reduce complexity of necessary relationships.
- By hiding implementation details, coupling is reduced.
- Apply the [Law of Demeter](#law-of-demeter).

Resources

- [Coupling (wikipedia.org)]()
- [Coupling And Cohesion (wiki.c2.com)](http://wiki.c2.com/?CouplingAndCohesion)

## Law of Demeter

Don't talk to strangers.

Why

- It usually tightens coupling
- It might reveal too much implementation details

How

A method of an object may only call methods of:

1. The object itself.
1. An argument of the method.
1. Any object created within the method.
1. Any direct properties/fields of the object.

Resources

- [Law of Demeter (wikipedia.org)](https://en.wikipedia.org/wiki/Law_of_Demeter)
- [The Law of Demeter Is Not A Dot Counting Exercise (haacked.com)](https://haacked.com/archive/2009/07/14/law-of-demeter-dot-counting.aspx/)

## Composition Over Inheritance

It is better to compose what an object can do than extend what it is. Compose
when there is a "has a" (or "uses a") relationship, inherit when "is a".

Why

- Less coupling between classes.
- Using inheritance, subclasses easily make assumptions, and break [LSP](#lsp).
- Increase flexibility: accommodate future requirements changes, otherwise
requiring restructuring of business-domain classes in the inheritance model.
- Avoid problems often associated with relatively minor changes to an
inheritance-based model including several generations of classes.

How

- Identify system object behaviors in separate interfaces, instead of creating a
hierarchical relationship to distribute behaviors among business-domain
classes via inheritance.
- Test for [LSP](#lsp) to decide when to inherit.

Resources

- [Composition over inheritance (wikipedia.org)](https://en.wikipedia.org/wiki/Composition_over_inheritance)
- [Favor Composition Over Inheritance (docs.microsoft.com)](https://docs.microsoft.com/en-us/archive/blogs/thalesc/favor-composition-over-inheritance)

## Orthogonality

> The basic idea of orthogonality is that things that are not related
> conceptually should not be related in the system.

Source: [Be Orthogonal](https://www.artima.com/intv/dry3.html)

> It is associated with simplicity; the more orthogonal the design, the fewer
> exceptions. This makes it easier to learn, read and write programs in a
> programming language. The meaning of an orthogonal feature is independent of
> context; the key parameters are symmetry and consistency.

Source:
[Orthogonality (wikipedia.org)]()

## Robustness Principle

> Be conservative in what you do, be liberal in what you accept from others

Collaborating services depend on each others interfaces. Often the interfaces
need to evolve causing the other end to receive unspecified data. A naive
implementation refuses to collaborate if the received data does not strictly
follow the specification. A more sophisticated implementation will still work
ignoring the data it does not recognize.

Why

- Ensure that a provider can make changes to support new demands, while causing
minimal breakage to their existing clients.

How

- Code that sends commands or data to other programs or machines should conform
completely to the specifications, but code that receives input should accept
non-conformant input as long as the meaning is clear.

Resources

- [Robustness Principle (wikipedia.org)](https://en.wikipedia.org/wiki/Robustness_principle)
- [Tolerant Reader (martinfowler.com)](https://martinfowler.com/bliki/TolerantReader.html)

## Inversion of Control

IoC inverts the flow of control. It's also known as the Hollywood Principle,
"Don't call us, we'll call you". A design principle in which custom-written
portions of a computer program receive the flow of control from a generic
framework. It carries the strong connotation that the reusable code and the
problem-specific code are developed independently even though they operate
together in an application.

Why

- Increase modularity of the program and make it extensible.
- To decouple the execution of a task from implementation.
- To focus a module on the task it is designed for.
- To free modules from assumptions about other systems and instead rely on
contracts.
- To prevent side effects when replacing a module.

How

- Using Factory pattern
- Using Service Locator pattern
- Using Dependency Injection
- Using contextualized lookup
- Using Template Method pattern
- Using Strategy pattern

Resources

- [Inversion of Control (wikipedia.org)](https://en.wikipedia.org/wiki/Inversion_of_control)
- [Inversion of Control Containers and the Dependency Injection pattern (martinfowler.com)](https://www.martinfowler.com/articles/injection.html)

## Maximise Cohesion

Cohesion of a single module or component is the degree to which its
responsibilities form a meaningful unit. Higher cohesion is better.

Why

- Reduced module complexity
- Increased system maintainability, because logical changes in the domain affect
fewer modules, and because changes in one module require fewer changes in
other modules.
- Increased module reusability, because application developers will find the
component they need more easily among the cohesive set of operations provided
by the module.

How

- Group related functionalities sharing a single responsibility (e.g. in a
module or class).

Resources

- [Cohesion (wikipedia.org)]()
- [Coupling And Cohesion (wiki.c2.com)](http://wiki.c2.com/?CouplingAndCohesion)

## Liskov Substitution Principle

The LSP is all about expected behavior of objects:

> Objects in a program should be replaceable with instances of their subtypes
> without altering the correctness of that program.

**LSP** is the L in [SOLID](#solid).

Resources

- [Liskov substitution principle (wikipedia.org)](https://en.wikipedia.org/wiki/Liskov_substitution_principle)
- [The Liskov Substitution Principle Explained (reflectoring.io)](https://reflectoring.io/lsp-explained/)

## Open/Closed Principle

Software entities (e.g. classes) should be open for extension, but closed for
modification. Such an entity can allow its behavior to be modified, without
altering its source code.

**Open/Closed** is the O in [SOLID](#solid).

Why

- Improve maintainability and stability by minimizing changes to existing code.

How

- Write classes that can be extended (as opposed to classes that can be
modified).
- Expose only the moving parts that need to change, hide everything else.

Resources

- [Open Closed Principle (wikipedia.org)](https://en.wikipedia.org/wiki/Open/closed_principle)
- [The Open Closed Principle (blog.cleancoder.com)](https://blog.cleancoder.com/uncle-bob/2014/05/12/TheOpenClosedPrinciple.html)

## Single Responsibility Principle

A class should never have more than one reason to change.

Long version: Every class should have a single responsibility, and that
responsibility should be entirely encapsulated by the class. Responsibility can
be defined as a reason to change, so a class or module should have one, and only
one, reason to change.

**SRP** is the S in [SOLID](#solid).

Why

- Maintainability: changes should be necessary only in one module or class.

How

- Apply [Curly's Law](#curlys-Law).

Resources

- [Single responsibility principle (wikipedia.org)](https://en.wikipedia.org/wiki/Single_responsibility_principle)

## Hide Implementation Details

A software module hides information (i.e. implementation details) by providing
an interface, and not leak any unnecessary information.

Why

- When the implementation changes, the interface clients are using does not have
to change.

How

- Minimize accessibility of classes and members.
- Don’t expose member data in public.
- Avoid putting private implementation details into a class’s interface.
- Decrease coupling to hide more implementation details.

Resources

- [Information hiding (wikipedia.org)](https://en.wikipedia.org/wiki/Information_hiding)

## Curly's Law

Curly's Law is about choosing a single, clearly defined goal for any particular
bit of code: Do One Thing.

- [Curly's Law: Do One Thing (blog.codinghorror.com)](https://blog.codinghorror.com/curlys-law-do-one-thing/)
- [The Rule of One or Curly’s Law (grsmentor.com)](http://grsmentor.com/blog/the-rule-of-one-or-curlys-law/)

## Encapsulate What Changes

A good design identifies the hotspots that are most likely to change and
encapsulates them behind an interface. When an anticipated change then occurs,
the modifications are kept local.

Why

- To minimize required modifications when a change occurs.

How

- Encapsulate the concept that varies behind an interface.
- Possibly separate the varying concept into its own module.

Resources

- [Encapsulate the Concept that Varies (principles-wiki.net)](http://principles-wiki.net/principles:encapsulate_the_concept_that_varies)
- [Encapsulate What Varies (blogs.msdn.microsoft.com)](https://blogs.msdn.microsoft.com/steverowe/2007/12/26/encapsulate-what-varies/)
- [Information hiding (wikipedia.org)](https://en.wikipedia.org/wiki/Information_hiding)

## Interface Segregation Principle

Reduce fat interfaces into multiple smaller and more specific client specific
interfaces. An interface should be more dependent on the code that calls it than
the code that implements it.

**ISP** is the I in [SOLID](#solid).

Why

- Keep a system decoupled and thus easier to refactor, change, and redeploy.

How

- Avoid fat interfaces. Classes should never have to implement methods that
violate the
[Single responsibility principle](#single-responsibility-principle).

Resources

- [Interface segregation principle (wikipedia.org)](https://en.wikipedia.org/wiki/Interface_segregation_principle)

## Command Query Separation

The Command Query Separation principle states that each method should be either
a command that performs an action, or a query that returns data to the caller
but not both. Asking a question should not modify the answer.

With this principle applied the programmer can code with much more confidence.
The query methods can be used anywhere and in any order since they do not mutate
the state. With commands one has to be more careful.

Why

- By clearly separating methods into queries and commands the programmer can
code with additional confidence without knowing each method's implementation
details.

How

- Implement each method as either a query or a command
- Apply naming convention to method names that implies whether the method is a
query or a command

Resources

- [Command Query Separation (wikipedia.org)](https://en.wikipedia.org/wiki/Command%E2%80%93query_separation)
- [Command Query Separation (martinfowler.com)](https://martinfowler.com/bliki/CommandQuerySeparation.html)

## Dependency Inversion Principle

Dependency Inversion is the strategy of depending upon interfaces or abstract
functions and classes rather than upon concrete functions and classes. Simply
put, when components of our system have dependencies, we don’t want to directly
inject a component’s dependency into another. Instead, we should use a level of
abstraction between them.

**DIP** is the D in [SOLID](#solid).

Why

- By decoupling the high-level modules from the low-level modules, the
high-level modules become more reusable and maintainable.
- Facilitates unit testing by allowing the use of mock objects, enabling
isolated testing of modules.
- Reduces the risk of breaking the system when changes are made.
- Allows adding new implementations without changing the existing code,
enhancing the system's extensibility.
- Concrete classes change frequently, while abstractions and interfaces change
much less.

How

- Define interfaces to capture behavior and use them to define the dependencies
of a class.
- Depends on abstractions, not on concretions.
- Using patterns like Factory, Service Locator, and Dependency Injection.
- Use the [Inversion of Control](#inversion-of-control) principle.

Resources

- [SOLID — Dependency Inversion Principle](https://medium.com/@inzuael/solid-dependency-inversion-principle-part-5-f5bec43ab22e)
- [Dependency inversion principle (wikipedia.org)](https://en.wikipedia.org/wiki/Dependency_inversion_principle)
- [System Design: Dependency Inversion Principle](https://www.baeldung.com/cs/dip)

## SOLID

A subset of programming principles:

- [Single Responsibility Principle](#single-responsibility-principle)
- [Open/Closed Principle](#openclosed-principle)
- [Liskov Substitution Principle](#liskov-substitution-principle)
- [Interface Segregation Principle](#interface-segregation-principle)
- [Dependency Inversion Principle](#dependency-inversion-principle)

## FIRST principles of testing

The FIRST testing principles mean that tests should be Fast, Isolated,
Repeatable, Self-validating and Timely.

Resources

- [F.I.R.S.T.](https://agileinaflash.blogspot.com/2009/02/first.html)

## Arrange, Act, Assert

The 3A's are a pattern to arrange and format code in unit tests. _Arrange_ all
necessary preconditions and inputs. _Act_ on the object or method under test.
_Assert_ that the expected results have occurred.

Resources

- [Arrange Act Assert (wiki.c2.com)](https://wiki.c2.com/?ArrangeActAssert)
- [3A - Arrange, Act, Assert (xp123.com)](https://xp123.com/articles/3a-arrange-act-assert/)