800===Dev Docs and License/Design Pattern

Design Patterns Introduced

블로글러 2024. 5. 28. 12:27

Design patterns in programming are like blueprints for solving common problems in software design, helping developers create more efficient and maintainable code by following proven solutions.


The Big Picture

Imagine you're building a house. Instead of inventing a new way to build every time, you follow blueprints that experienced architects have used successfully. Similarly, in programming, design patterns are like these blueprints. They provide general reusable solutions to common problems that arise in software design.

Core Concepts

  1. Definition of Design Patterns:

    • Design patterns are typical solutions to common problems in software design.
    • They are templates designed to help write code that is easy to understand and reuse.
  2. Categories of Design Patterns:

    • Creational Patterns: Deal with object creation mechanisms, trying to create objects in a manner suitable to the situation.
      • Examples: Singleton, Factory, Builder, Prototype.
    • Structural Patterns: Deal with object composition, or how objects and classes are composed to form larger structures.
      • Examples: Adapter, Composite, Proxy, Flyweight, Facade.
    • Behavioral Patterns: Deal with communication between objects, defining how they interact and how responsibilities are distributed among them.
      • Examples: Observer, Strategy, Command, State, Visitor.

Detailed Walkthrough

Let's dive into a few examples to understand these patterns better:

  1. Singleton Pattern (Creational):

    • Ensures a class has only one instance and provides a global point of access to it.

    • Analogy: Think of it as the sole president of a country. There can only be one president at a time, and everyone knows who it is.

    • Example:

      class Singleton:
          _instance = None
      
          @staticmethod
          def get_instance():
              if Singleton._instance is None:
                  Singleton._instance = Singleton()
              return Singleton._instance
  2. Observer Pattern (Behavioral):

    • Defines a one-to-many dependency between objects so that when one object changes state, all its dependents are notified and updated automatically.

    • Analogy: Consider a newspaper subscription. When the newspaper (subject) publishes a new edition, all subscribers (observers) receive a copy.

    • Example:

      class Observer:
          def update(self, message):
              pass
      
      class Subject:
          def __init__(self):
              self._observers = []
      
          def add_observer(self, observer):
              self._observers.append(observer)
      
          def remove_observer(self, observer):
              self._observers.remove(observer)
      
          def notify_observers(self, message):
              for observer in self._observers:
                  observer.update(message)
      
      class ConcreteObserver(Observer):
          def update(self, message):
              print("Received message:", message)
      
      # Usage
      subject = Subject()
      observer1 = ConcreteObserver()
      observer2 = ConcreteObserver()
      
      subject.add_observer(observer1)
      subject.add_observer(observer2)
      
      subject.notify_observers("New Update Available!")
  3. Adapter Pattern (Structural):

    • Allows incompatible interfaces to work together by acting as a bridge between two incompatible interfaces.

    • Analogy: Think of an electric plug adapter that lets you plug a device from one country into the wall socket of another.

    • Example:

      class EuropeanSocket:
          def voltage(self):
              return 230
      
      class AmericanSocket:
          def voltage(self):
              return 120
      
      class Adapter:
          def __init__(self, socket):
              self._socket = socket
      
          def voltage(self):
              return self._socket.voltage()
      
      # Usage
      euro_socket = EuropeanSocket()
      adapter = Adapter(euro_socket)
      print(adapter.voltage())  # Outputs: 230

Understanding Through an Example

Imagine you are developing a notification system for an application. Using the Observer pattern, you can create a flexible system where different parts of the application can subscribe to certain notifications and be updated automatically when events occur. This decouples the sender from the receivers, promoting a more modular and maintainable design.

Conclusion and Summary

Design patterns provide a proven template for solving common problems in software design. They are categorized into creational, structural, and behavioral patterns, each addressing different aspects of software development. By understanding and applying these patterns, developers can create more robust, maintainable, and scalable applications.

Test Your Understanding

  1. What is the primary purpose of the Singleton pattern?
  2. How does the Observer pattern help in decoupling components in a software system?
  3. Can you think of a real-world scenario where the Adapter pattern would be useful?

Reference

For further reading and more examples, you can refer to:

  • "Design Patterns: Elements of Reusable Object-Oriented Software" by Erich Gamma, Richard Helm, Ralph Johnson, and John Vlissides (the "Gang of Four" book).
  • Refactoring.Guru on Design Patterns
728x90

'800===Dev Docs and License > Design Pattern' 카테고리의 다른 글

더 나은 Java 코드 리팩토링 가이드 🛠️  (0) 2024.11.13
Observer Pattern with Java  (0) 2024.05.30
Clean Architecture  (0) 2024.05.28
Singleton Pattern with Java  (0) 2024.05.28
Factory Pattern with Java  (0) 2024.05.27