Strategy Design Pattern Example
Let’s consider a sorting application where we need to sort a list of integers. However, the sorting algorithm to be used may vary depending on factors such as the size of the list and the desired performance characteristics.
Challenges Without Using Strategy Pattern:
- Limited Flexibility: Implementing sorting algorithms directly within the main sorting class can make the code inflexible. Adding new sorting algorithms or changing existing ones would require modifying the main class, which violates the Open/Closed Principle.
- Code Duplication: Without a clear structure, you may end up duplicating sorting logic to handle different algorithms. This can lead to maintenance issues and inconsistency in the system.
- Hard-Coded Logic: Implementing sorting logic directly within the main sorting class can make the code rigid and difficult to extend or modify. Making changes to the sorting algorithm becomes cumbersome and error-prone.
How Strategy Pattern helps to solve above challenges :
The Strategy Design Pattern addresses these challenges by encapsulating each sorting algorithm into separate classes. This allows for better organization, code reuse, and flexibility in the system. Here’s how the Strategy Pattern helps:
- Code Reusability: By encapsulating sorting algorithms into separate strategy classes, you can reuse these strategies across different parts of the system. This reduces code duplication and promotes maintainability.
- Flexibility and Extensibility: With the Strategy Pattern, you can easily add new sorting algorithms or change existing ones without modifying existing code. Each strategy is independent and can be swapped or extended without affecting other parts of the system.
- Separation of Concerns: The Strategy Pattern promotes a clean separation of concerns by isolating sorting logic into separate strategy classes. This improves code readability, testability, and maintainability.
Complete Code of the above example:
1. Context(SortingContext)
Java
public class SortingContext { private SortingStrategy sortingStrategy; public SortingContext(SortingStrategy sortingStrategy) { this .sortingStrategy = sortingStrategy; } public void setSortingStrategy(SortingStrategy sortingStrategy) { this .sortingStrategy = sortingStrategy; } public void performSort( int [] array) { sortingStrategy.sort(array); } } |
2. Strategy Interface(SortingStrategy)
Java
public interface SortingStrategy { void sort( int [] array); } |
3. Concrete Strategies
Java
// BubbleSortStrategy public class BubbleSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Bubble Sort algorithm System.out.println( "Sorting using Bubble Sort" ); } } // MergeSortStrategy public class MergeSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Merge Sort algorithm System.out.println( "Sorting using Merge Sort" ); } } // QuickSortStrategy public class QuickSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Quick Sort algorithm System.out.println( "Sorting using Quick Sort" ); } } |
4. Client Component
Java
public class Client { public static void main(String[] args) { // Create SortingContext with BubbleSortStrategy SortingContext sortingContext = new SortingContext( new BubbleSortStrategy()); int [] array1 = { 5 , 2 , 9 , 1 , 5 }; sortingContext.performSort(array1); // Output: Sorting using Bubble Sort // Change strategy to MergeSortStrategy sortingContext.setSortingStrategy( new MergeSortStrategy()); int [] array2 = { 8 , 3 , 7 , 4 , 2 }; sortingContext.performSort(array2); // Output: Sorting using Merge Sort // Change strategy to QuickSortStrategy sortingContext.setSortingStrategy( new QuickSortStrategy()); int [] array3 = { 6 , 1 , 3 , 9 , 5 }; sortingContext.performSort(array3); // Output: Sorting using Quick Sort } } |
Complete code for the above example
Below is the complete code for the above example:
Java
// SortingContext.java class SortingContext { private SortingStrategy sortingStrategy; public SortingContext(SortingStrategy sortingStrategy) { this .sortingStrategy = sortingStrategy; } public void setSortingStrategy(SortingStrategy sortingStrategy) { this .sortingStrategy = sortingStrategy; } public void performSort( int [] array) { sortingStrategy.sort(array); } } // SortingStrategy.java interface SortingStrategy { void sort( int [] array); } // BubbleSortStrategy.java class BubbleSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Bubble Sort algorithm System.out.println( "Sorting using Bubble Sort" ); // Actual Bubble Sort Logic here } } // MergeSortStrategy.java class MergeSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Merge Sort algorithm System.out.println( "Sorting using Merge Sort" ); // Actual Merge Sort Logic here } } // QuickSortStrategy.java class QuickSortStrategy implements SortingStrategy { @Override public void sort( int [] array) { // Implement Quick Sort algorithm System.out.println( "Sorting using Quick Sort" ); // Actual Quick Sort Logic here } } // Client.java public class Client { public static void main(String[] args) { // Create SortingContext with BubbleSortStrategy SortingContext sortingContext = new SortingContext( new BubbleSortStrategy()); int [] array1 = { 5 , 2 , 9 , 1 , 5 }; sortingContext.performSort(array1); // Output: Sorting using Bubble Sort // Change strategy to MergeSortStrategy sortingContext.setSortingStrategy( new MergeSortStrategy()); int [] array2 = { 8 , 3 , 7 , 4 , 2 }; sortingContext.performSort(array2); // Output: Sorting using Merge Sort // Change strategy to QuickSortStrategy sortingContext.setSortingStrategy( new QuickSortStrategy()); int [] array3 = { 6 , 1 , 3 , 9 , 5 }; sortingContext.performSort(array3); // Output: Sorting using Quick Sort } } |
Output
Sorting using Bubble Sort Sorting using Merge Sort Sorting using Quick Sort |
Strategy Design Pattern
The Strategy Design Pattern defines a family of algorithms, encapsulates each one, and makes them interchangeable, allowing clients to switch algorithms dynamically without altering the code structure.
Important Topics for the Strategy Design Pattern
- What is the Strategy Design Pattern?
- Characteristics of the Strategy Design Pattern?
- Components of the Strategy Design Pattern
- Communication between the Components
- Real-World Analogy of Strategy Design Pattern
- Strategy Design Pattern Example
- When to use the Strategy Design Pattern?
- When not to use the Strategy Design Pattern?
- Advantages of the Strategy Design Pattern
- Disadvantages of the Strategy Design Pattern