디시인사이드 갤러리

갤러리 이슈박스, 최근방문 갤러리

갤러리 본문 영역

최근에 각잡고 짠 구현 코드인데 평가 바람

ㅆㅇㅆ(124.216) 2025.07.30 14:03:27
조회 104 추천 0 댓글 3

using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;

namespace JDWCodeVault.Patterns.Creational
{
    /// <summary>
    /// 플루언트 빌더 패턴의 기본 인터페이스
    /// 메소드 체이닝을 통한 객체 생성을 지원합니다.
    /// </summary>
    /// <typeparam name="T">생성할 객체의 타입</typeparam>
    public interface IObjectBuilder<T>
    {
        /// <summary>
        /// 속성 값을 설정합니다.
        /// </summary>
        /// <typeparam name="TProperty">속성의 타입</typeparam>
        /// <param name="property">설정할 속성</param>
        /// <param name="value">설정할 값</param>
        /// <returns>빌더 인스턴스 (메소드 체이닝용)</returns>
        IObjectBuilder<T> Configure<TProperty>(Expression<Func<T, TProperty>> property, TProperty value);
       
        /// <summary>
        /// 조건부로 속성 값을 설정합니다.
        /// </summary>
        /// <typeparam name="TProperty">속성의 타입</typeparam>
        /// <param name="condition">설정 조건</param>
        /// <param name="property">설정할 속성</param>
        /// <param name="value">설정할 값</param>
        /// <returns>빌더 인스턴스 (메소드 체이닝용)</returns>
        IObjectBuilder<T> ConfigureIf<TProperty>(bool condition, Expression<Func<T, TProperty>> property, TProperty value);
       
        /// <summary>
        /// 설정된 값들로 최종 객체를 생성합니다.
        /// </summary>
        /// <returns>생성된 객체</returns>
        T Build();
       
        /// <summary>
        /// 빌더를 초기 상태로 리셋합니다.
        /// </summary>
        /// <returns>리셋된 빌더 인스턴스</returns>
        IObjectBuilder<T> Reset();
    }

    /// <summary>
    /// 제네릭 플루언트 빌더 구현체
    /// 리플렉션을 사용하여 동적으로 객체를 생성합니다.
    /// </summary>
    /// <typeparam name="T">생성할 객체의 타입</typeparam>
    public class FluentBuilder<T> : IObjectBuilder<T> where T : class, new()
    {
        private readonly Dictionary<string, object> properties;
        private readonly Dictionary<string, Func<object, bool>> validators;

        public FluentBuilder()
        {
            properties = new Dictionary<string, object>();
            validators = new Dictionary<string, Func<object, bool>>();
        }

        public IObjectBuilder<T> Configure<TProperty>(Expression<Func<T, TProperty>> property, TProperty value)
        {
            var propertyName = GetPropertyName(property);
            properties[propertyName] = value;
            return this;
        }

        public IObjectBuilder<T> ConfigureIf<TProperty>(bool condition, Expression<Func<T, TProperty>> property, TProperty value)
        {
            if (condition)
            {
                Configure(property, value);
            }
            return this;
        }

        /// <summary>
        /// 속성에 대한 검증 규칙을 추가합니다.
        /// </summary>
        /// <typeparam name="TProperty">속성의 타입</typeparam>
        /// <param name="property">검증할 속성</param>
        /// <param name="validator">검증 함수</param>
        /// <returns>빌더 인스턴스</returns>
        public FluentBuilder<T> AddValidator<TProperty>(Expression<Func<T, TProperty>> property, Func<TProperty, bool> validator)
        {
            var propertyName = GetPropertyName(property);
            validators[propertyName] = obj => validator((TProperty)obj);
            return this;
        }

        public T Build()
        {
            ValidateProperties();
           
            var instance = new T();
            var type = typeof(T);

            foreach (var kvp in properties)
            {
                var propertyInfo = type.GetProperty(kvp.Key);
                if (propertyInfo != null && propertyInfo.CanWrite)
                {
                    propertyInfo.SetValue(instance, kvp.Value);
                }
            }

            return instance;
        }

        public IObjectBuilder<T> Reset()
        {
            properties.Clear();
            return this;
        }

        private void ValidateProperties()
        {
            foreach (var kvp in properties)
            {
                if (validators.TryGetValue(kvp.Key, out var validator))
                {
                    if (!validator(kvp.Value))
                    {
                        throw new ArgumentException($"Validation failed for property: {kvp.Key}");
                    }
                }
            }
        }

        private static string GetPropertyName<TProperty>(Expression<Func<T, TProperty>> property)
        {
            if (property.Body is MemberExpression memberExpression)
            {
                return memberExpression.Member.Name;
            }
            throw new ArgumentException("Expression must be a property access", nameof(property));
        }
    }

    /// <summary>
    /// 단계별 빌더 패턴을 위한 타입 안전한 빌더
    /// 컴파일 타임에 필수 속성 설정을 강제합니다.
    /// </summary>
    /// <typeparam name="T">생성할 객체의 타입</typeparam>
    public class StepBuilder<T> where T : class
    {
        protected readonly Dictionary<string, object> properties;

        protected StepBuilder()
        {
            properties = new Dictionary<string, object>();
        }

        protected StepBuilder(Dictionary<string, object> existingProperties)
        {
            properties = new Dictionary<string, object>(existingProperties);
        }

        protected TNext CreateNext<TNext>() where TNext : StepBuilder<T>, new()
        {
            var next = new TNext();
            foreach (var kvp in properties)
            {
                next.properties[kvp.Key] = kvp.Value;
            }
            return next;
        }
    }

    /// <summary>
    /// 불변 객체 생성을 위한 빌더
    /// 빌드 시점에 모든 속성이 설정된 불변 객체를 생성합니다.
    /// </summary>
    /// <typeparam name="T">생성할 불변 객체의 타입</typeparam>
    public class ImmutableBuilder<T> : IObjectBuilder<T> where T : class
    {
        private readonly Dictionary<string, object> properties;
        private readonly Func<Dictionary<string, object>, T> factory;

        public ImmutableBuilder(Func<Dictionary<string, object>, T> factory)
        {
            this.factory = factory ?? throw new ArgumentNullException(nameof(factory));
            properties = new Dictionary<string, object>();
        }

        public IObjectBuilder<T> Configure<TProperty>(Expression<Func<T, TProperty>> property, TProperty value)
        {
            var propertyName = GetPropertyName(property);
            properties[propertyName] = value;
            return this;
        }

        public IObjectBuilder<T> ConfigureIf<TProperty>(bool condition, Expression<Func<T, TProperty>> property, TProperty value)
        {
            if (condition)
            {
                Configure(property, value);
            }
            return this;
        }

        public T Build()
        {
            return factory(new Dictionary<string, object>(properties));
        }

        public IObjectBuilder<T> Reset()
        {
            properties.Clear();
            return this;
        }

        private static string GetPropertyName<TProperty>(Expression<Func<T, TProperty>> property)
        {
            if (property.Body is MemberExpression memberExpression)
            {
                return memberExpression.Member.Name;
            }
            throw new ArgumentException("Expression must be a property access", nameof(property));
        }
    }

    /// <summary>
    /// 빌더 팩토리 - 다양한 빌더 타입을 생성합니다.
    /// </summary>
    public static class BuilderFactory
    {
        /// <summary>
        /// 플루언트 빌더를 생성합니다.
        /// </summary>
        /// <typeparam name="T">생성할 객체의 타입</typeparam>
        /// <returns>플루언트 빌더 인스턴스</returns>
        public static FluentBuilder<T> CreateFluent<T>() where T : class, new()
        {
            return new FluentBuilder<T>();
        }

        /// <summary>
        /// 불변 객체 빌더를 생성합니다.
        /// </summary>
        /// <typeparam name="T">생성할 불변 객체의 타입</typeparam>
        /// <param name="factory">객체 생성 팩토리 함수</param>
        /// <returns>불변 객체 빌더 인스턴스</returns>
        public static ImmutableBuilder<T> CreateImmutable<T>(Func<Dictionary<string, object>, T> factory) where T : class
        {
            return new ImmutableBuilder<T>(factory);
        }
    }
} 평가 부탁드립니다.

추천 비추천

0

고정닉 0

0

댓글 영역

전체 댓글 0
본문 보기

하단 갤러리 리스트 영역

왼쪽 컨텐츠 영역

갤러리 리스트 영역

갤러리 리스트
번호 제목 글쓴이 작성일 조회 추천
설문 순간의 말실수로 이미지 타격이 큰 것 같은 스타는? 운영자 25/09/01 - -
AD 가전디지털, 신학기 페스타! 운영자 25/08/29 - -
2881831 트럼프의 달러 무기화 속에 전세계는 지금 달러와 이혼 중 발명도둑잡기갤로그로 이동합니다. 08.16 126 0
2881830 음기 충전 발명도둑잡기갤로그로 이동합니다. 08.16 92 0
2881829 영화 <웨폰> 인기래서 생각나는 예전 글 발명도둑잡기갤로그로 이동합니다. 08.16 80 0
2881828 유심없는 폰으로 테러글 올리면 어케잡냐는애 있었는데 밀우갤로그로 이동합니다. 08.16 94 0
2881827 DAVICHI '그걸 사랑이라고 말하지마' 발명도둑잡기갤로그로 이동합니다. 08.16 63 0
2881824 <기억의 미래> 발명도둑잡기갤로그로 이동합니다. 08.16 75 0
2881823 오늘의 작사 실마리: '트'란 제목의 노래 발명도둑잡기갤로그로 이동합니다. 08.16 90 0
2881821 음양론에 따른 노래 구분 발명도둑잡기갤로그로 이동합니다. 08.16 63 0
2881820 "새벽 5시40분 입장" 장범준, '미라클 모닝' 공연한다 발명도둑잡기갤로그로 이동합니다. 08.16 80 0
2881807 관심경제의 부작용과 자본주의 관계 발명도둑잡기갤로그로 이동합니다. 08.16 83 0
2881801 나는요. 24년 2월 24일 탈당했었어요. [1] 넥도리아(175.196) 08.16 131 0
2881800 요즘 코딩 배운다는게 이런거냐 [2] ㅇㅇ(220.89) 08.16 155 0
2881799 하루에 3경조줄 감사의 코딩 공기역학갤로그로 이동합니다. 08.16 126 0
2881798 스프링5 처음 시작하면 추천할만한 글 프갤러(121.133) 08.15 126 0
2881796 세대론, '노인을 위한 나라는 없다' 관련 생각나는 예전 글 발명도둑잡기갤로그로 이동합니다. 08.15 124 0
2881793 오늘 김치찜 먹었는데 맛있더라 ㅇㅅㅇㅋㅋ ㅇㅇ(223.39) 08.15 65 0
2881792 방금 전 트위터 자동 추천 발명도둑잡기갤로그로 이동합니다. 08.15 97 0
2881790 오늘의 소설, 영화 실마리: CIA 신고열풍 신고했다 간첩으로 포섭 [2] 발명도둑잡기갤로그로 이동합니다. 08.15 98 0
2881785 에구궁 허리야.. ♥냥덩이♥갤로그로 이동합니다. 08.15 100 0
2881783 챗티너무 느려서 코파일럿 제미나이쓰게된다 [2] 헬마스터갤로그로 이동합니다. 08.15 133 0
2881782 근데 지금 2030세대가 아버지 세대보다 [1] ㅇㅇ(211.234) 08.15 105 0
2881781 ♥냥덩이♥갤로그로 이동합니다. 08.15 73 0
2881779 일하는거 존나 쉬운데 프갤러(121.139) 08.15 109 1
2881778 변영주 감독 <낮은 목소리> 발명도둑잡기갤로그로 이동합니다. 08.15 80 0
2881777 광복과 이재명 [1] 발명도둑잡기갤로그로 이동합니다. 08.15 122 0
2881776 Java 전문가 마스터 봐라. [1] 프갤러(59.16) 08.15 144 0
2881773 양양 후기 ♥냥덩이♥갤로그로 이동합니다. 08.15 136 0
2881772 여름바다는 확실히 습하당 ♥냥덩이♥갤로그로 이동합니다. 08.15 94 0
2881771 도와주세용ㅠ 11갤로그로 이동합니다. 08.15 71 0
2881770 나님이 마시는 술⭐+ ♥냥덩이♥갤로그로 이동합니다. 08.15 86 0
2881769 유튜브 뮤직 알고리즘 너무 구리더라 [2] 헬마스터갤로그로 이동합니다. 08.15 102 0
2881766 윤미향 비난의 상당수 원인은 성매수, 성추행에 대한 죄책감 때문이다 발명도둑잡기갤로그로 이동합니다. 08.15 70 0
2881765 신봉선과 냥덩이의 결혼을 허가합니다 [2] 발명도둑잡기갤로그로 이동합니다. 08.15 87 0
2881764 오늘의 소설 영화 실마리: FBI, CIA 등에 마약 수사로 포섭된 래퍼 발명도둑잡기갤로그로 이동합니다. 08.15 74 0
2881763 게임 개발 공부하려고 프리서버 뜯어보고 있는데 ㅇㅇ갤로그로 이동합니다. 08.15 112 0
2881762 AI 나오고 느끼는 것이 AI가 최고의 스승이다. [2] ㅆㅇㅆ(124.216) 08.15 136 0
2881761 전광훈 현상의 뿌리를 파헤치다 [기자의 추천 책] 발명도둑잡기갤로그로 이동합니다. 08.15 55 0
2881757 요즘 느끼는 것이 그냥 하면 해볼만하더라 [1] ㅆㅇㅆ찡갤로그로 이동합니다. 08.15 107 0
2881756 여름밤바다⭐+✨☀+⚡+☄++ ♥냥덩이♥갤로그로 이동합니다. 08.15 57 0
2881755 술먹고도 안피곤하단걸 자랑으로 생각하는 애들이 있네... ㅇㅇ(223.39) 08.15 67 0
2881753 ai 창작러로써 배운거... ㅇㅇ(106.102) 08.15 89 0
2881752 경상도 여자가 예쁨 [1] ♥냥덩이♥갤로그로 이동합니다. 08.15 111 0
2881750 어제 맥주 2리터먹고 오늘 하루종일 잠잤음 ㅇㅅㅇ;; ㅇㅇ(223.39) 08.15 97 0
2881749 뮤지컬 우키시마마루 봤다 발명도둑잡기갤로그로 이동합니다. 08.15 87 0
2881748 유다 배신 비슷한 설화가 근방 신화에 더 먼저 있었나 발명도둑잡기갤로그로 이동합니다. 08.15 90 0
2881747 개발자 면접 썰.txt [2] 프갤러(58.29) 08.15 477 1
2881746 바이브 코딩으로 프로그램 만드는대 프갤러(45.130) 08.15 93 0
2881745 노인을 혐오 비하하면 왜 안됨??노인새끼들이 먼저 2030 갈라치기선동하 뒷통수한방(1.213) 08.15 61 0
2881738 You are being Watched.. [1] ♥냥덩이♥갤로그로 이동합니다. 08.15 110 0
2881733 좇센살이 ) 노력하면 평범하게 살수있음 뒷통수한방(1.213) 08.15 81 0
뉴스 강유미 “한번 갔다 왔다”…9살 아들 사진 공개 디시트렌드 09.04
갤러리 내부 검색
제목+내용게시물 정렬 옵션

오른쪽 컨텐츠 영역

실시간 베스트

1/8

뉴스

디시미디어

디시이슈

1/2