Jump to content

Dependency inversion principle

From Wikipedia, the free encyclopedia

This is an old revision of this page, as edited by Klodr (talk | contribs) at 02:03, 19 January 2015 (DIP Implementations). The present address (URL) is a permanent link to this revision, which may differ significantly from the current revision.

In object-oriented programming, the dependency inversion principle refers to a specific form of decoupling software modules. When following this principle, the conventional dependency relationships established from high-level, policy-setting modules to low-level, dependency modules are inverted (i.e. reversed), thus rendering high-level modules independent of the low-level module implementation details. The principle states:[1]

A. High-level modules should not depend on low-level modules. Both should depend on abstractions.
B. Abstractions should not depend on details. Details should depend on abstractions.

The principle inverts the way some people may think about object-oriented design, dictating that both high- and low-level objects must depend on the same abstraction.[2]

Traditional Layers Pattern

In conventional application architecture, lower-level components are designed to be consumed by higher-level components which enable increasingly complex systems to be built. In this composition, higher-level components depend directly upon lower-level components to achieve some task. This dependency upon lower-level components limits the reuse opportunities of the higher-level components.[3]

An interpretation of the traditional layers architectural pattern.

The goal of the dependency inversion principle is to avoid this highly coupled distribution with the mediation of an abstract layer, and to increase the re-usability of higher/policy layers.

Ownership Inversion

With the addition of an abstract layer, both high- and lower-level layers avoid the traditional dependencies from top to bottom. Nevertheless the ″inversion″ concept doesn't mean that lower-level layers depends instead of higher-level layers. Both layers should depend on abstractions that draw the behavior needed by higher-level layers.

A simple application of DIP.

In a direct application of dependency inversion, the abstracts are owned by the upper/policy layers. This architecture groups in the same package the higher/policy components with the abstracts that define lower services. The lower-level layers are created by inheritance/implementation of these abstracts classes or interfaces.[4]

The inversion of the dependencies and ownership encourages the re-usability of the higher/policy layers. Upper layers could use other implementations of the lower services. When the lower-level layer components are closed or when the application requires the reuse of existing services, is common that an Adapter mediates between the services and the abstractions.

Abstraction Dependency

The presence of abstractions to accomplish DIP have other design implications in an Object Oriented program:

  • All member variables in a class must be interfaces or abstracts.
  • All concrete class packages must connect only through interface/abstract classes packages.
  • No class should derive from a concrete class.
  • No method should override an implemented method.[5]
  • All variable instantiation requires the implementation of a Creational pattern as the Factory Method or the Factory pattern, or the more complex use of a Dependency Injection framework.

DIP Implementations

Two common implementations of DIP use similar logical architecture, with different implications.

A direct implementation packages the policy classes with service abstracts classes in one library. In this implementation high-level components and low-level components are distributed into separate packages/libraries, where interfaces defining the behavior/services required by the high-level component are owned by, and exist within the high-level component's library. The implementation of the high-level component's interface by the low level component requires that the low-level component package depend upon the high-level component for compilation, thus inverting the conventional dependency relationship.

Figures 1 and 2 illustrate code with the same functionality, however in figure 2, an interface has been used to invert the dependency. The direction of dependency can be chosen to maximize policy code reuse, and eliminate cyclic dependencies.

In this version of DIP, the dependency to the higher-level layers difficult the re-utilization of lower layer components. This implementation instead ″inverts″ the traditional dependency —from top to bottom— to the opposite from bottom to top.

A more flexible solution separate in three packages/libraries the solution:

DIP, separate layers in their own packages/libraries.

The separation of all layers in their own package encourage re-utilization of any layer, providing robustness and mobility.[6]

Applying the dependency inversion principle can also be seen as applying the Adapter pattern, i.e. the high-level class defines its own adapter interface which is the abstraction that the other high-level classes depend on. The adaptee implementation also depends on the adapter interface abstraction (of course, since it implements its interface) while it can be implemented by using code from within its own low-level module. The high-level has no dependency to the low-level module since it only uses the low-level indirectly through the adapter interface by invoking polymorphic methods to the interface which are implemented by the adaptee and its low-level module.

Various patterns such as Plugin, Service Locator, or Dependency Injection are employed to facilitate the run-time provisioning of the chosen low-level component implementation to the high-level component.

History

The dependency inversion principle was postulated by Robert C. Martin and described in several publications including the paper Object Oriented Design Quality Metrics: an analysis of dependencies,[7] an article appearing in the C++ Report in May 1996 entitled The Dependency Inversion Principle,[8] and the books Agile Software Development, Principles, Patterns, and Practices, and Agile Principles, Patterns, and Practices in C#.

See also

References

  1. ^ Martin, Robert C. (2003). Agile Software Development, Principles, Patterns, and Practices. Prentice Hall. p. 127. ISBN 978-0135974445.
  2. ^ Freeman, Eric; Freeman, Elisabeth; Kathy, Sierra; Bert, Bates (2004). Hendrickson, Mike; Loukides, Mike (eds.). "Head First Design Patterns" (paperback). 1. O'REILLY. ISBN 978-0-596-00712-6. Retrieved 2012-06-21. {{cite journal}}: Cite journal requires |journal= (help)
  3. ^ Martin, Robert C. Agile Software Development, Principles, Patterns, and Practices. Prentice Hall. p. 128. ISBN 978-0135974445.
  4. ^ Martin, Robert C. Agile Software Development, Principles, Patterns, and Practices. Prentice Hall. p. 127-128. ISBN 978-0135974445.
  5. ^ Martin, Robert C. Agile Software Development, Principles, Patterns, and Practices. Prentice Hall. p. 129. ISBN 978-0135974445.
  6. ^ Martin, Robert C. Agile Software Development, Principles, Patterns, and Practices. Prentice Hall. p. 131. ISBN 978-0135974445.
  7. ^ Object Oriented Design Quality Metrics: an analysis of dependencies Robert C. Martin, C++ Report, Sept/Oct 1995
  8. ^ The Dependency Inversion Principle, Robert C. Martin, C++ Report, May 1996