소프트웨어 엔지니어로서 우리는 유지 관리가 가능하고 유연하며 확장 가능한 시스템을 만드는 작업을 지속적으로 수행하고 있습니다. 이러한 맥락에서 디자인 패턴은 구조화되고 재사용 가능한 방식으로 반복되는 문제를 해결하는 데 도움이 되는 강력한 도구입니다. 그러한 디자인 패턴 중 하나가 행동 패턴 계열에 속하는 전략 패턴
입니다.전략 패턴을 사용하면 알고리즘 계열을 정의하고 각 알고리즘을 캡슐화하여 상호 교환 가능하게 만들 수 있습니다. 이는 클라이언트가 시스템의 핵심 기능을 변경하지 않고도 런타임에 적절한 알고리즘이나 전략을 선택할 수 있음을 의미합니다.
이 블로그에서는 전략 패턴, 핵심 개념과 구성 요소, 실제 사례, 그리고 이를 사용해야 하는 시기와 이유에 대해 자세히 알아보겠습니다. 또한 전략 패턴이 추상화, 열거형, 심지어 팩토리 패턴과 함께 작동하여 디자인을 더욱 강력하고 유연하게 만드는 방법도 살펴보겠습니다.
전략 패턴은 알고리즘의 동작을 런타임 시 선택할 수 있도록 하는 동작 디자인 패턴입니다. 단일하고 모놀리식 알고리즘을 사용하는 대신 전략 패턴을 사용하면 동작(또는 전략)을 상호 교환할 수 있으므로 시스템이 더욱 유연해지고 유지 관리가 쉬워집니다.
전략 패턴은 다음과 같은 경우에 특히 유용합니다.
관심사 분리: 전략 패턴은 알고리즘의 관심사를 시스템의 나머지 부분과 분리합니다. 클라이언트 코드는 알고리즘이 내부적으로 어떻게 작동하는지 인식하지 못하므로 더욱 모듈화됩니다.
확장성: 기존 코드를 변경하지 않고 새로운 전략 클래스만 추가하면 새로운 알고리즘을 추가할 수 있습니다.
유지관리성: 개별 전략 클래스에 다양한 동작을 위임하여 코드의 복잡성을 줄여 유지관리를 더 쉽게 만듭니다.
간단한 알고리즘: 작업 중인 알고리즘이 간단하고 변경되지 않는 경우 전략 패턴을 사용하는 것은 과잉일 수 있습니다.
전략이 너무 많음: 전략이 많으면 클래스가 폭발적으로 늘어나 가독성이 떨어지고 복잡성이 증가할 수 있습니다.
간헐적인 변경: 알고리즘이 자주 변경되지 않는 경우 전략 패턴을 도입하면 불필요한 복잡성이 발생할 수 있습니다.
전략 패턴은 다음과 같은 주요 구성 요소로 구성됩니다.
컨텍스트:
전략:
구체적인 전략:
사용자가 신용카드, 페이팔, 암호화폐 등 다양한 방법으로 결제할 수 있는 결제 처리 시스템을 생각해 보겠습니다. 결제 방식에 따라 결제가 처리되는 방식이 다르지만, 각 결제 방식의 세부 사항을 고려하지 않고 컨텍스트(이 경우 ShoppingCart)에서 결제를 처리할 수 있어야 합니다.
먼저 열거형을 사용하여 다양한 결제 방법을 정의하겠습니다. 이렇게 하면 결제 방법 선택이 안전하고 관리하기 쉬워집니다.
public enum PaymentMethod { CREDIT_CARD, PAYPAL, CRYPTOCURRENCY; }
이 클래스는 결제 처리에 필요한 세부정보를 요약합니다. 여기에는 결제 방법 및 결제 세부정보(예: 카드번호, 이메일, 암호화폐 주소)가 포함됩니다.
public class PaymentInformation { private PaymentMethod paymentMethod; private String paymentDetails; public PaymentInformation(PaymentMethod paymentMethod, String paymentDetails) { this.paymentMethod = paymentMethod; this.paymentDetails = paymentDetails; } public PaymentMethod getPaymentMethod() { return paymentMethod; } public String getPaymentDetails() { return paymentDetails; } }
이는 모든 결제 전략의 기본 인터페이스가 됩니다. 모든 구체적인 전략이 구현되는 공통 메소드 pay()를 정의합니다.
public enum PaymentMethod { CREDIT_CARD, PAYPAL, CRYPTOCURRENCY; }
여기에서는 CreditCardPayment, PayPalPayment 및 CryptoPayment에 대한 구체적인 전략을 구현합니다. 각 클래스는 결제 유형에 따라 pay() 메소드를 구현합니다.
public class PaymentInformation { private PaymentMethod paymentMethod; private String paymentDetails; public PaymentInformation(PaymentMethod paymentMethod, String paymentDetails) { this.paymentMethod = paymentMethod; this.paymentDetails = paymentDetails; } public PaymentMethod getPaymentMethod() { return paymentMethod; } public String getPaymentDetails() { return paymentDetails; } }
public abstract class PaymentStrategy { protected PaymentInformation paymentInformation; public PaymentStrategy(PaymentInformation paymentInformation) { this.paymentInformation = paymentInformation; } public abstract void pay(double amount); protected boolean validatePaymentDetails() { return paymentInformation != null && paymentInformation.getPaymentDetails() != null && !paymentInformation.getPaymentDetails().isEmpty(); } }
public class CreditCardPayment extends PaymentStrategy { public CreditCardPayment(PaymentInformation paymentInformation) { super(paymentInformation); } @Override public void pay(double amount) { if (validatePaymentDetails()) { System.out.println("Paid " + amount + " using Credit Card: " + paymentInformation.getPaymentDetails()); } else { System.out.println("Invalid Credit Card details."); } } }
팩토리 패턴을 사용하여 결제 수단에 따라 적절한 결제 전략을 인스턴스화하겠습니다. 이를 통해 시스템이 더욱 유연해지고 클라이언트가 런타임에 결제 방법을 선택할 수 있습니다.
public class PayPalPayment extends PaymentStrategy { public PayPalPayment(PaymentInformation paymentInformation) { super(paymentInformation); } @Override public void pay(double amount) { if (validatePaymentDetails()) { System.out.println("Paid " + amount + " using PayPal: " + paymentInformation.getPaymentDetails()); } else { System.out.println("Invalid PayPal details."); } } }
ShoppingCart 클래스는 결제 전략이 사용되는 컨텍스트입니다. 공장에서 선택한 전략에 결제 책임을 위임합니다.
public class CryptoPayment extends PaymentStrategy { public CryptoPayment(PaymentInformation paymentInformation) { super(paymentInformation); } @Override public void pay(double amount) { if (validatePaymentDetails()) { System.out.println("Paid " + amount + " using Cryptocurrency to address: " + paymentInformation.getPaymentDetails()); } else { System.out.println("Invalid cryptocurrency address."); } } }
public class PaymentStrategyFactory { public static PaymentStrategy createPaymentStrategy(PaymentInformation paymentInformation) { switch (paymentInformation.getPaymentMethod()) { case CREDIT_CARD: return new CreditCardPayment(paymentInformation); case PAYPAL: return new PayPalPayment(paymentInformation); case CRYPTOCURRENCY: return new CryptoPayment(paymentInformation); default: throw new IllegalArgumentException("Unsupported payment method: " + paymentInformation.getPaymentMethod()); } } }
public class ShoppingCart { private PaymentStrategy paymentStrategy; public ShoppingCart(PaymentInformation paymentInformation) { this.paymentStrategy = PaymentStrategyFactory.createPaymentStrategy(paymentInformation); } public void checkout(double amount) { paymentStrategy.pay(amount); } public void setPaymentInformation(PaymentInformation paymentInformation) { this.paymentStrategy = PaymentStrategyFactory.createPaymentStrategy(paymentInformation); } }
핵심 로직을 수정하지 않고도 동작이 변경됩니다.
전략 패턴은 시스템의 유연성과 모듈성을 달성하기 위한 필수 디자인 패턴입니다. 알고리즘을 캡슐화하는 우아한 방법을 제공하고 기존 코드를 수정하지 않고도 런타임 유연성을 활성화합니다. 결제 처리 시스템, 정렬 알고리즘 라이브러리 또는 게임 AI 엔진을 구축하는 경우 전략 패턴은 요구 사항이 발전함에 따라 코드를 더욱 유지 관리하고 확장 가능하며 수정하기 쉽게 만드는 데 도움이 될 수 있습니다.
추상화, 열거형 및 팩토리 패턴을 활용하면 유형이 안전하고 유연한 더욱 강력한 시스템을 구축할 수 있습니다.
위 내용은 전략 디자인 패턴 익히기: 개발자를 위한 가이드의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!