Reasons:
Degree in which parts of the code are dependent on each other. Low coupling means independent modules/functios etc. Helps in making the parts more reusable. Easier to update. There is always going to be coupling as parts make use of each other. The point is to be aware and decide where and how to create the coupling. To keep (some) control over maintainability of the software. Its a strategic effort to choose where you want to have what type of coupling.
- strong type of coupling, try to avoid! - content coupling is when a function updates something directly which is passed in as an argument. (modifies something / state of something outside of its own scope) - problem is that coupling occurs on low level implementation between classes, so the acting function or class needs to know/depends on the internal implementation of the other class.
Can be avoided somewhat by creating functions (setters getters) so that the internal parameters are "encapsulated".
- When modules interact with each other and methods share data through parameters. Is a “good” type of coupling as it is a low level data.
principle of least knowledge, units should only talk or interact with closely related units.
Assign a function to the class that can perform it best (as in has the data to be able to do it).
- General Responsibility Assignment Software Patterns (GRASP)
Is for type checking only, the actual implementation is still just done in one function. I guess these days since you can define multiple input types and return types with type hints i guess this decorator may be considered obsolete.
Inheritence is very tight coupling. Should be used with care as complex inheritance hierachies can make maintence very complicated. Try not to go deeper than 1 level of inheritance.
Concept similar to interfaces, its a contract for sub classes.
Defines a interface by defining the properties and functions a class can adhere to. Its more close to duck typing. Does not rely on inheritance. In runtime the interpreter checks if all the members match.
Protocols allow for finer granularity (interface segregation), more protocols can be combined in a class before it becomes messy as there is less tight coupling in code.
Python supports multiple inheritance. So a class can have multiple parent classes. The method resolution order (left to right by defined clases) is important to take into account as it defines which method in parent class gets called if the same method exists in multiple classes.
The multiple inheritance feature is also named mixins as you mix multiple types of classes actually multiple inheritance tree's together.
When using mixins you need to know the implementation details of classes to figure out a resolution order that will work. So the class structure can become complex and hard to work with.
Arjan advises against using mixins, and when used the parent classes should only have methods so that initialisation issues (think of resolution order) with parameters are avoided.
Mixin classes should not have instance variables.
So its powerfull but try to avoid and use with care as it will get complicated.
According to Python Mixin (Pythontutorial.net) a mixin class can be seen as the interface concept but with implementation. “A mixin bundles a set of methods for reuse”