[객체지향] 7.유닛 테스트와 테스트 주도 개발(TDD), 테스트 주도 개발(TDD)의 장점과 실용적인 접근 방법

1. 유닛 테스트란?

유닛 테스트는 소프트웨어의 각 단위, 즉 함수나 메서드 등의 독립적인 개별 구성 요소가
기대한 대로 작동하는지 확인하는 과정입니다. 이는 일반적으로 자동화된 테스트로,
코드의 사소한 변경에도 프로그램의 다른 부분이 영향을 받지 않는지 확인하는 데 필수적입니다.
C#에서는 다양한 프레임워크를 사용하여 유닛 테스트를 작성할 수 있으며, NUnit
xUnit이 가장 널리 사용되는 도구입니다.

2. 테스트 주도 개발(TDD)란?

테스트 주도 개발(Test-Driven Development, TDD)은 소프트웨어 개발 프로세스
중 하나로, 실제 코드를 구현하기 전에 테스트 코드를 작성하는 접근 방법입니다.
TDD는 다음의 세 가지 단계로 이루어집니다:

  1. Red: 테스트를 먼저 작성하고, 이는 당연히 실패합니다.
  2. Green: 실패한 테스트를 통과 할 수 있도록 최소한의 코드 작성.
  3. Refactor: 통과하는 코드를 리팩토링 하여 코드 품질 개선.

3. TDD의 장점

TDD는 여러 가지 장점을 제공합니다:

  • 코드 품질 향상: 코드가 명확하게 정의된 테스트를 통과해야 하므로,
    품질이 개선됩니다.
  • 디버깅 용이: 버그가 발생할 경우, 어느 테스트가 실패했는지 확인하여
    문제를 쉽게 찾을 수 있습니다.
  • 명세서 역할: 테스트는 요구사항을 명확히 문서화하므로,
    이해 관계자와 개발자 간의 소통이 원활해집니다.
  • 리팩토링 지원: 코드 리팩토링 시, 기존 테스트가 문제없이 통과하는지
    확인할 수 있어 자신감을 갖고 코드를 개선할 수 있습니다.

4. 실용적인 접근 방법: TDD 적용하기

TDD를 효과적으로 적용하기 위해서는 몇 가지 실용적인 전략이 필요합니다.

4.1 간단한 요구사항 정의

TDD의 첫 번째 단계는 요구사항을 명확하게 정의하는 것입니다. 예를 들어,
간단한 계산기를 만든다고 가정합시다. 덧셈 기능을 먼저 요구사항으로 정합니다.
그 후, 해당 덧셈 기능에 대한 테스트를 작성합니다.

4.2 테스트 코드 작성

다음은 C#에서 NUnit을 사용해 간단한 덧셈 메서드에 대한 테스트입니다:


using NUnit.Framework;

[TestFixture]
public class CalculatorTests
{
    [Test]
    public void Add_TwoPositiveNumbers_ReturnsSum()
    {
        // Arrange
        var calculator = new Calculator();
        
        // Act
        var result = calculator.Add(2, 3);
        
        // Assert
        Assert.AreEqual(5, result);
    }
}

4.3 코드 작성

다음 단계에서는 요구사항을 만족하기 위해 실제 코드를 작성합니다.
위의 테스트를 통과시키기 위해, 간단한 Calculator 클래스를 작성합니다.


public class Calculator
{
    public int Add(int a, int b)
    {
        return a + b;
    }
}

4.4 리팩토링

코드가 작동하고 테스트를 통과했다면, 리팩토링을 통해 코드를 개선합니다.
여러 메서드를 추가하거나 코드를 정리하여 가독성을 높일 수 있습니다.

5. TDD의 어려움과 극복 방법

TDD를 적용하는 과정에서 몇 가지 어려움이 발생할 수 있습니다:

  • 시간 소모: 테스트를 먼저 작성해야 하기 때문에 개발 속도가 느려질 수 있습니다.
    그러나 장기적으로는 코드 품질이 향상되어 시간 절약이 가능합니다.
  • 복잡한 요구사항: 요구사항이 복잡할 경우, 테스트 작성이 어려울 수 있습니다.
    이 경우, 기능을 여러 개의 조각으로 나누어 쉽게 테스트할 수 있도록 합니다.

6. TDD 도구와 리소스

다음은 TDD를 지원하는 유용한 도구 및 리소스입니다:

  • NUnit: C#을 위한 모든 플랫폼에서 유닛 테스트를 작성할 수 있는 프레임워크.
  • xUnit: 모던 C# 개발에 적합한 유닛 테스트 프레임워크.
  • Moq: 테스트에서 의존성을 주입할 수 있는 모킹 라이브러리.

7. 결론

TDD는 소프트웨어 개발 과정에서 코드 품질을 향상시키고, 버그를 예방하며,
요구사항에 대한 명확한 문서 역할을 합니다. 개발자는 TDD를 통해 자신감을 가지고
코드를 작성하며 빠르게 변화하는 요구사항에 적응할 수 있습니다.
오늘 배운 내용을 바탕으로 TDD를 실무에 적용해 보시기 바랍니다.

[객체지향] 2.C# 최신 문법과 기능 활용, switch 표현식과 패턴 매칭

C#은 현재 소프트웨어 개발에서 매우 인기가 높고 강력한 프로그래밍 언어입니다. 특히, C#의 최신 버전에서는 많은 기능이 추가되어
개발자들이 코드 작성 시 더 효율적으로 작업할 수 있게 해줍니다. 이 글에서는 C#의 최신 문법과 기능 중, 특히 switch 표현식과
패턴 매칭에 대해 자세히 알아보겠습니다.

1. switch 표현식 소개

switch 표현식은 C# 8.0에서 도입된 기능으로, 기존의 switch 문보다 더욱 간결하고 명확한 문법을 제공합니다. 이 표현식은
조건에 따라 값을 반환하며, 결과를 변수에 할당할 수 있는 장점이 있습니다. 기본적인 구문은 다음과 같습니다.

C#
// switch 표현식의 기본 예제
int number = 2;
string result = number switch
{
    1 => "하나",
    2 => "둘",
    3 => "셋",
    _ => "기타" // 기본(default)값
};
Console.WriteLine(result); // 출력: 둘

1.1. switch 표현식의 장점

switch 표현식은 가독성이 뛰어나며, 코드가 간결해져서 유지보수하기 쉬운 장점이 있습니다. 또한, 각 케이스에서 반환할 값을
바로 정의할 수 있으므로, 기존의 switch 문보다 코드 흐름을 이해하기 쉬워집니다.

2. 패턴 매칭

C# 7.0부터 도입된 패턴 매칭은 객체의 형을 검사하고, 일치하는 형에 따라 작업을 수행할 수 있게 합니다. 이는 코드의 유연성과
가독성을 높이는 데 기여합니다. 패턴 매칭은 여러형으로 나누어 사용할 수 있으며, 여기에서는 switch 표현식 내에서
어떻게 활용될 수 있는지 살펴보겠습니다.

2.1. 기본 패턴 매칭

C#
// 기본 패턴 매칭 예제
object obj = "Hello";
string message = obj switch
{
    string s when s.StartsWith("H") => "H로 시작하는 문자열입니다.",
    int i => "정수입니다: " + i,
    _ => "알 수 없는 타입입니다."
};
Console.WriteLine(message); // 출력: H로 시작하는 문자열입니다.

2.2. 더 복잡한 패턴 사용

C#의 패턴 매칭은 타입 패턴, 값 패턴, 널 패턴 등을 지원합니다. 이러한 패턴은 복잡한 조건을 명확하게 표현할 수 있게 해줍니다.
아래 예제를 살펴봅시다.

C#
// 다양한 패턴 매칭 예제
object data = new List { 1, 2, 3 };

string resultPattern = data switch
{
    int[] arr => $"정수 배열입니다. 길이: {arr.Length}",
    List list when list.Count > 0 => $"정수 리스트입니다. 첫 번째 요소: {list[0]}",
    _ => "알 수 없는 데이터 타입입니다."
};
Console.WriteLine(resultPattern); // 출력: 정수 리스트입니다. 첫 번째 요소: 1

3. switch 표현식과 패턴 매칭의 결합

switch 표현식과 패턴 매칭은 함께 사용되어 훨씬 강력한 기능을 발휘할 수 있습니다. 코드의 가독성을 높이고, 유지보수성을
향상시키는 데 큰 도움이 됩니다. 다음은 이 두 가지 기능을 결합한 예제입니다.

C#
// switch 표현식과 패턴 매칭의 결합 예제
object dataObject = 42;
string resultCombined = dataObject switch
{
    int n when n > 0 => $"{n}는 양수입니다.",
    int n when n < 0 => $"{n}는 음수입니다.",
    null => "값이 없습니다.",
    _ => "알 수 없는 타입입니다."
};
Console.WriteLine(resultCombined); // 출력: 42는 양수입니다.

4. 결론

C#의 최신 문법과 기능, 특히 switch 표현식과 패턴 매칭을 활용하면 더욱 간결하고 가독성이 높은 코드를 작성할 수 있습니다.
이러한 기능들은 개발자들이 더 적은 코드로 더 많은 일을 할 수 있도록 도와주며, 현대 소프트웨어 개발의 요구 사항에
잘 부합합니다. 실제 프로젝트에서 이러한 기능을 사용함으로써 코드의 구조를 개선하고, 개발 생산성을 높이는 데
기여할 수 있습니다. C#을 사용하는 모든 개발자들에게 이러한 기능들을 적극적으로 활용해 보기를 권장합니다.

참고 자료

[객체지향] 10.Reflection과 고급 메타프로그래밍, 의존성 주입에서 Reflection 사용 사례

글쓴이: 조광형

날짜: 2024년 11월 26일

1. 서론

현대 소프트웨어 개발에서 객체지향 프로그래밍(OOP)과 메타프로그래밍은 필수적인 개념으로 자리 잡고 있습니다. C#은 이러한 개념을 지원하는 여러 강력한 기능을 제공합니다. 특히, Reflection은 클래스, 메서드, 속성 등의 메타데이터를 런타임에 동적으로 확인하고 조작할 수 있는 기능을 제공합니다. 본 글에서는 Reflection과 고급 메타프로그래밍에 대해 살펴보고, 의존성 주입 패턴에서 Reflection의 사용 사례를 상세히 논의하겠습니다.

2. Reflection이란?

Reflection은 C#에서 제공하는 기능으로, 프로그램의 메타데이터를 런타임에 확인하고 조작할 수 있게 해줍니다. 이 기능을 통해 개발자는 클래스의 구조, 속성, 메서드 등을 동적으로 탐색하고, 이를 기반으로 동적인 객체 생성을 하거나 메서드를 호출할 수 있습니다.

2.1 Reflection의 주요 구성 요소

Reflection은 주로 다음과 같은 클래스와 네임스페이스로 구성됩니다:

  • System.Reflection: Reflection 관련 기능을 포함하는 네임스페이스.
  • Assembly: 어셈블리에 대한 정보 및 메타데이터를 제공.
  • Type: 클래스, 인터페이스, 열거형 등 객체의 유형 정보 제공.
  • MethodInfo: 메서드에 대한 메타데이터 정보 제공.
  • PropertyInfo: 속성에 대한 메타데이터 정보 제공.

3. Reflection의 사용 예시

Reflection의 기본적인 사용법을 이해하기 위해, 다음은 간단한 C# 클래스와 해당 클래스를 Reflection을 통해 탐색하는 예제입니다.


using System;
using System.Reflection;

public class SampleClass
{
    public int Number { get; set; }
    public string Text { get; set; }

    public void Display()
    {
        Console.WriteLine($"Number: {Number}, Text: {Text}");
    }
}

class Program
{
    static void Main()
    {
        Type type = typeof(SampleClass);
        Console.WriteLine($"클래스 이름: {type.Name}");

        PropertyInfo[] properties = type.GetProperties();
        foreach (var property in properties)
        {
            Console.WriteLine($"속성: {property.Name} - 타입: {property.PropertyType}");
        }

        MethodInfo method = type.GetMethod("Display");
        Console.WriteLine($"메서드 이름: {method.Name}");
    }
}
            

위 예제에서 SampleClass의 메타데이터에 접근하여 해당 클래스의 속성과 메서드 정보를 출력할 수 있습니다.

4. 메타프로그래밍의 개념과 활용

메타프로그래밍은 프로그램을 작성하는 것과 같은 언어의 기법을 사용하여 프로그램 그 자체를 조작하는 프로그래밍 패러다임입니다. C#의 Reflection을 활용하면 메타프로그래밍을 통해 코드의 재사용성과 유연성을 높일 수 있습니다. 다음은 메타프로그래밍을 활용한 예제입니다.


using System;

public class DynamicLoader
{
    public T Load(string typeName) where T : class
    {
        Type type = Type.GetType(typeName);
        if (type == null) throw new InvalidOperationException("타입을 찾을 수 없습니다.");

        return Activator.CreateInstance(type) as T;
    }
}

class Program
{
    static void Main()
    {
        DynamicLoader loader = new DynamicLoader();
        var myClassInstance = loader.Load("Namespace.MyClass");

        // myClassInstance를 사용할 수 있습니다.
    }
}
            

이 예제에서 DynamicLoader는 타입 이름을 문자열로 받아 해당 타입의 인스턴스를 동적으로 생성하는 역할을 합니다. 이를 통해 코드에서 의존성을 줄이고, 더 유연한 설계를 할 수 있습니다.

5. 의존성 주입에서 Reflection의 사용 사례

의존성 주입(Dependency Injection, DI)은 객체의 의존 관계를 외부에서 주입함으로써 결합도를 낮추고, 테스트 용이성을 높이는 디자인 패턴입니다. Reflection은 DI 컨테이너에서 유형을 동적으로 해결하는 데 필수적으로 사용됩니다.

5.1 의존성 주입의 개념

의존성 주입은 객체가 직접 의존하는 객체를 생성하지 않고, 외부에서 주입받는 방식으로, 이를 통해 객체 간의 결합도를 낮추어 유연한 구조를 제공합니다.

5.2 DI 컨테이너에서 Reflection 활용하기

DI 컨테이너를 구현하기 위해 Reflection을 사용할 수 있습니다. 다음은 간단한 DI 컨테이너의 예제입니다:


using System;
using System.Collections.Generic;
using System.Reflection;

public class SimpleContainer
{
    private readonly Dictionary _registrations = new Dictionary();

    public void Register() where T : I
    {
        _registrations[typeof(I)] = typeof(T);
    }

    public I Resolve()
    {
        Type typeToResolve = _registrations[typeof(I)];
        ConstructorInfo constructor = typeToResolve.GetConstructors()[0];
        ParameterInfo[] parameters = constructor.GetParameters();

        object[] parameterInstances = new object[parameters.Length];
        for (int i = 0; i < parameters.Length; i++)
        {
            parameterInstances[i] = Resolve(parameters[i].ParameterType);
        }

        return (I)constructor.Invoke(parameterInstances);
    }
}

// 사용 예
public interface IService { void Execute(); }
public class Service : IService { public void Execute() => Console.WriteLine("Service called"); }
public class Client
{
    private readonly IService _service;

    public Client(IService service) => _service = service;

    public void Run() => _service.Execute();
}

class Program
{
    static void Main()
    {
        SimpleContainer container = new SimpleContainer();
        container.Register();
        Client client = container.Resolve();
        client.Run();
    }
}
            

위의 예제에서 SimpleContainer는 타입을 등록하고, 이를 바탕으로 생성자를 통해 의존성을 해결합니다. Reflection을 사용하여 적절한 생성자를 찾고, 생성자의 파라미터를 재귀적으로 해결합니다.

6. Reflection 사용의 장단점

6.1 장점

  • 유연성: 런타임에 클래스를 동적으로 탐색하고 사용할 수 있게 해줍니다.
  • 코드 재사용성: 메타프로그래밍 기법을 통해 중복 코드를 줄일 수 있습니다.

6.2 단점

  • 성능 저하: Reflection은 일반적인 메서드 호출보다 느리며, 런타임 성능에 영향을 줄 수 있습니다.
  • 타입 안전성 감소: 컴파일 타임에 타입이 결정되지 않으므로, 런타임 오류가 발생할 수 있습니다.

7. 결론

Reflection과 메타프로그래밍은 C#의 뛰어난 기능으로, 전통적인 프로그래밍 모델을 넘어서는 유연하고 동적인 프로그래밍을 가능하게 합니다. 특히, 의존성 주입 패턴에서 Reflection은 강력한 도구로 활용됩니다. 그러나 개발자는 Reflection 사용의 장단점을 충분히 이해하고, 필요에 따라 적절하게 활용해야 합니다. 이를 통해 보다 효과적이고 유지 보수성이 높은 코드를 작성할 수 있습니다.

이 글이 C#의 Reflection과 고급 메타프로그래밍에 대한 이해를 높이는 데 도움이 되었기를 바랍니다.

[객체지향] 1.객체지향 프로그래밍의 핵심 원칙, 의존성 주입과 그 역할

객체지향 프로그래밍(Object-Oriented Programming, OOP)은 소프트웨어 개발에서 널리 사용되는 프로그래밍 패러다임입니다. OOP의 목표는 소프트웨어의 구조를 명확히 하고, 코드의 재사용성을 극대화하며, 유지 보수를 용이하게 하기 위함입니다. 이 글에서는 OOP의 핵심 원칙과 의존성 주입(Dependency Injection, DI)의 역할에 대해 깊이 있게 살펴보겠습니다.

1. 객체지향 프로그래밍의 핵심 원칙

OOP의 핵심 원칙은 다음과 같습니다:

  • 캡슐화(Encapsulation): 객체의 내부 상태를 보호하고, 객체의 행동을 정의하는 원칙입니다. 이 원칙은 데이터와 메소드를 하나의 단위로 묶고, 외부에서는 내부 구현을 알 필요 없도록 합니다.
  • 상속(Inheritance): 새로운 클래스가 기존 클래스의 특성을 재사용할 수 있는 기능입니다. 코드를 재사용할 수 있게 해주며, 클래스 간의 계층 구조를 형성합니다.
  • 다형성(Polymorphism): 동일한 인터페이스를 통해 다른 객체를 처리할 수 있는 능력입니다. 코드의 유연성을 높이고, 유지보수를 용이하게 합니다.
  • 추상화(Abstraction): 필요한 정보만을 노출하여 복잡성을 줄이는 것입니다. 복잡한 시스템을 단순화시켜 개발자와 사용자 모두에게 이해하기 쉽게 만듭니다.

1.1 캡슐화의 예

public class BankAccount {
    private decimal balance;

    public void Deposit(decimal amount) {
        balance += amount;
    }

    public void Withdraw(decimal amount) {
        if (amount > balance) throw new InvalidOperationException("Insufficient funds.");
        balance -= amount;
    }

    public decimal GetBalance() {
        return balance;
    }
}

1.2 상속의 예

public class Animal {
    public virtual void Speak() {
        Console.WriteLine("Animal sound");
    }
}

public class Dog : Animal {
    public override void Speak() {
        Console.WriteLine("Bark");
    }
}

1.3 다형성의 예

public void MakeAnimalSpeak(Animal animal) {
    animal.Speak();
}

MakeAnimalSpeak(new Dog()); // Bark

1.4 추상화의 예

public abstract class Shape {
    public abstract double Area();
}

public class Circle : Shape {
    public double Radius { get; set; }

    public override double Area() {
        return Math.PI * Radius * Radius;
    }
}

2. 의존성 주입(Dependency Injection)

의존성 주입은 소프트웨어 디자인 패턴 중 하나로, 객체가 다른 객체에 의존하는 방식을 관리하는 기법입니다. DI는 객체의 생성과 의존성 관리를 외부로 분리하여, 코드의 결합도를 낮추고, 테스트 용이성을 높입니다. DI의 주된 목표는 클래스가 다른 클래스와의 의존성을 줄여, 결합도를 낮추는 것입니다.

2.1 의존성 주입의 중요성

의존성 주입의 중요성은 다음과 같습니다:

  • **유지보수 용이성**: 객체 간의 관계가 명확해지므로, 코드의 변경이 용이합니다.
  • **테스트 용이성**: Mock 객체를 사용해 의존성 주입을 통해 단위 테스트를 쉽게 수행할 수 있습니다.
  • **재사용성 증가**: 독립적인 객체로 설계됨으로써 코드의 재사용성이 증가합니다.
  • **유연성 및 확장성**: 새로운 기능 추가나 변경이 쉬워집니다.

2.2 의존성 주입의 종류

의존성 주입의 종류는 다음과 같습니다:

  • 생성자 주입(Constructor Injection): 의존하는 객체를 생성자의 매개변수로 전달합니다.
  • 설정자 주입(Setter Injection): 의존하는 객체를 setter 메서드를 통해 주입합니다.
  • 인터페이스 주입(Interface Injection): 의존하는 객체에 주입 메서드를 정의한 인터페이스를 구현하게 합니다.

2.3 생성자 주입의 예

public class PaymentService {
    private readonly IPaymentGateway paymentGateway;

    public PaymentService(IPaymentGateway paymentGateway) {
        this.paymentGateway = paymentGateway;
    }

    public void ProcessPayment(decimal amount) {
        paymentGateway.Process(amount);
    }
}

2.4 설정자 주입의 예

public class NotificationService {
    private INotificationSender notificationSender;

    public void SetNotificationSender(INotificationSender sender) {
        notificationSender = sender;
    }

    public void SendNotification(string message) {
        notificationSender.Send(message);
    }
}

2.5 인터페이스 주입의 예

public interface IHasDependency {
    void InjectDependency(IDependency dependency);
}

public class SomeService : IHasDependency {
    private IDependency dependency;

    public void InjectDependency(IDependency dependency) {
        this.dependency = dependency;
    }
}

3. 의존성 주입 프레임워크

많은 현대 C# 애플리케이션에서 DI를 구현하기 위해 다양한 DI 프레임워크를 사용합니다. 이러한 프레임워크는 객체의 생명주기 관리, 의존성 해소, 스코프 관리 등 다양한 기능을 제공합니다.

다음은 C#에서 인기 있는 의존성 주입 프레임워크입니다:

  • Autofac: 확장성과 유연성이 뛰어난 강력한 DI 컨테이너입니다.
  • Unity: Microsoft에서 개발된 DI 컨테이너로, 간편한 설정을 제공합니다.
  • Ninject: 플러그인처럼 사용할 수 있는 다양한 기능이 있는 DI 컨테이너입니다.
  • ASP.NET Core DI: ASP.NET Core의 기본 내장 DI 프레임워크로, 간단한 설정으로 DI를 제공합니다.

3.1 ASP.NET Core DI의 예

public class Startup {
    public void ConfigureServices(IServiceCollection services) {
        services.AddScoped();
        services.AddScoped();
    }
}

public class HomeController : Controller {
    private readonly PaymentService paymentService;

    public HomeController(PaymentService paymentService) {
        this.paymentService = paymentService;
    }
}

4. 결론

객체지향 프로그래밍(OOP)은 소프트웨어 개발의 핵심 원칙을 제공하여 코드의 이해와 유지 보수를 용이하게 합니다. 의존성 주입(Dependency Injection)은 이러한 OOP 원칙을 더욱 강화시켜, 결합도를 낮추고, 소프트웨어의 품질을 높이는 중요한 기법입니다. 이러한 기법들을 이해하고 활용하는 것은 개발자에게 많은 이점을 제공합니다.

이 글을 통해 객체지향 프로그래밍의 핵심 원칙과 의존성 주입에 대한 깊은 이해를 돕고자 하였으며, 실제 코드 예제를 통해 적용 방법을 제시하였습니다. 앞으로의 프로젝트에서 OOP와 DI를 통해 더욱 견고한 소프트웨어를 개발하시기 바랍니다.

[객체지향] 3.디자인 패턴 개요 및 구현 예제, 생성 패턴 싱글턴, 팩토리 메서드, 추상 팩토리

1. 디자인 패턴 소개

디자인 패턴은 소프트웨어 엔지니어링에서 반복적으로 발생하는 문제를 해결하기 위해
고안된 일반적인 솔루션을 의미합니다. 이러한 패턴은 개발자들이 더 나은 구조의
소프트웨어를 설계할 수 있도록 도와줍니다. 디자인 패턴은 주로 객체지향 프로그래밍에서
널리 사용되며, 코드를 재사용하고 유지관리할 수 있는 용이성을 제공합니다.

디자인 패턴은 크게 세 가지 유형으로 나눌 수 있습니다: 생성 패턴, 구조 패턴, 행동 패턴.
이번 글에서는 생성 패턴에 대해 자세히 살펴보겠습니다.

2. 생성 패턴

생성 패턴은 객체 생성 관련 문제를 다룹니다. 이 패턴들은 객체 생성 방식을 정의하여
클라이언트 코드와 객체 생성 로직 간의 결합도를 낮춰줍니다. 생성 패턴의 대표적인 예로는
싱글턴(Singleton), 팩토리 메서드(Factory Method), 추상 팩토리(Abstract Factory)
가 있습니다.

3. 싱글턴 패턴

싱글턴 패턴은 클래스의 인스턴스가 오직 하나만 존재하도록 보장하며,
그 인스턴스에 접근할 수 있는 전역적인 접근점을 제공합니다.
이 패턴은 주로 설정, 로그 기록, 데이터베이스 연결 등과 같이
애플리케이션 전역에서 단일 인스턴스가 필요한 경우에 사용됩니다.

3.1. 구현 예제


public class Singleton
{   
    private static Singleton instance;

    private Singleton()
    {
    }

    public static Singleton Instance
    {
        get
        {
            if (instance == null)
            {
                instance = new Singleton();
            }
            return instance;
        }
    }

    public void SomeBusinessLogic()
    {
        // 비즈니스 로직
    }
}
        

설명: 위의 구현에서 Singleton 클래스는 private 생성자를 가지고 있어
외부에서 직접 인스턴스를 생성할 수 없습니다. Instance 속성을 통해 싱글턴 객체를
접근할 수 있으며, 인스턴스가 null인 경우에만 새로 생성됩니다.

3.2. 사용 예제


class Program
{
    static void Main(string[] args)
    {
        Singleton singleton = Singleton.Instance;
        singleton.SomeBusinessLogic();
    }
}
        

설명: Main 메서드에서는 Singleton.Instance를 호출하여
싱글턴 인스턴스에 접근하고, 비즈니스 로직을 수행합니다.

4. 팩토리 메서드 패턴

팩토리 메서드 패턴은 객체 생성의 인터페이스를 정의하지만,
어떤 클래스의 인스턴스를 만들지는 서브클래스에서 결정하도록 하는
패턴입니다. 이 패턴은 객체 생성을 캡슐화하여 클라이언트 코드에서
객체 생성에 대한 의존성을 줄여줍니다.

4.1. 구현 예제


public abstract class Creator
{
    public abstract Product FactoryMethod();

    public void SomeOperation()
    {
        // 제품 객체를 생성하고 사용함
        var product = FactoryMethod();
    }
}

public class ConcreteCreatorA : Creator
{
    public override Product FactoryMethod()
    {
        return new ConcreteProductA();
    }
}

public class ConcreteCreatorB : Creator
{
    public override Product FactoryMethod()
    {
        return new ConcreteProductB();
    }
}

public abstract class Product
{
    public abstract string GetInfo();
}

public class ConcreteProductA : Product
{
    public override string GetInfo()
    {
        return "ConcreteProductA";
    }
}

public class ConcreteProductB : Product
{
    public override string GetInfo()
    {
        return "ConcreteProductB";
    }
}
        

설명: Creator 추상 클래스는 FactoryMethod() 메서드를 정의합니다.
ConcreteCreatorA와 ConcreteCreatorB는 각각 다른 종류의 Product 객체를 생성합니다.

4.2. 사용 예제


class Program
{
    static void Main(string[] args)
    {
        Creator creator;

        // A 타입의 제품 생성
        creator = new ConcreteCreatorA();
        var productA = creator.FactoryMethod();
        Console.WriteLine(productA.GetInfo());

        // B 타입의 제품 생성
        creator = new ConcreteCreatorB();
        var productB = creator.FactoryMethod();
        Console.WriteLine(productB.GetInfo());
    }
}
        

설명: Main 메서드에서는 ConcreteCreatorA와 ConcreteCreatorB를 사용하여
서로 다른 타입의 제품을 생성하고 출력합니다.

5. 추상 팩토리 패턴

추상 팩토리 패턴은 관련된 객체들의 집합을 생성하는 인터페이스를 제공합니다.
이 패턴은 클라이언트 코드가 구체적인 클래스에 의존하지 않고,
고수준의 인터페이스를 통해 객체를 생성할 수 있도록 합니다.
주로 관련된 제품군을 만들어야 할 때 유용합니다.

5.1. 구현 예제


public interface IAbstractFactory
{
    IProductA CreateProductA();
    IProductB CreateProductB();
}

public class ConcreteFactory1 : IAbstractFactory
{
    public IProductA CreateProductA()
    {
        return new ProductA1();
    }

    public IProductB CreateProductB()
    {
        return new ProductB1();
    }
}

public class ConcreteFactory2 : IAbstractFactory
{
    public IProductA CreateProductA()
    {
        return new ProductA2();
    }

    public IProductB CreateProductB()
    {
        return new ProductB2();
    }
}

public interface IProductA
{
    string GetProductInfo();
}

public class ProductA1 : IProductA
{
    public string GetProductInfo()
    {
        return "ProductA1";
    }
}

public class ProductA2 : IProductA
{
    public string GetProductInfo()
    {
        return "ProductA2";
    }
}

public interface IProductB
{
    string GetProductInfo();
}

public class ProductB1 : IProductB
{
    public string GetProductInfo()
    {
        return "ProductB1";
    }
}

public class ProductB2 : IProductB
{
    public string GetProductInfo()
    {
        return "ProductB2";
    }
}
        

설명: IAbstractFactory 인터페이스는 두 가지 타입의 제품을
생성하는 메서드를 정의합니다. 각 구체적인 팩토리는 이를 구현하여
스스로의 제품을 생성합니다.

5.2. 사용 예제


class Program
{
    static void Main(string[] args)
    {
        IAbstractFactory factory = new ConcreteFactory1();
        var productA = factory.CreateProductA();
        var productB = factory.CreateProductB();

        Console.WriteLine(productA.GetProductInfo());
        Console.WriteLine(productB.GetProductInfo());
    }
}
        

설명: Main 메서드에서는 ConcreteFactory1을 통해
ProductA1과 ProductB1 객체를 생성하고 정보를 출력합니다.
디펜던시가 구체적인 제품이 아닌 추상화된 형태로 제공되므로,
변경이 용이해집니다.

6. 결론

이번 글에서는 C#에서 디자인 패턴, 특히 생성 패턴에 대해 알아보았습니다.
싱글턴, 팩토리 메서드, 추상 팩토리 패턴을 통해 객체 생성을 좀 더 유연하고,
재사용 가능한 방법으로 처리할 수 있음을 보여주었습니다. 디자인 패턴을 적절히
활용하면 코드의 가독성과 유지보수성을 높일 수 있습니다. 앞으로도 다양한
디자인 패턴에 대한 이해를 넓히고, 적절히 활용해 보시길 바랍니다.