Microsoft MVP성태의 닷넷 이야기
리플렉션 성능 관련 질문 드립니다. [링크 복사], [링크+제목 복사],
조회: 14548
글쓴 사람
popo
홈페이지
첨부 파일
 

안녕하세요 다음과 같이 DataTable을 특정 모델 클래스의 List<T>타입으로 Convert해서 사용하고 있습니다.

그런데 성능상 데이터가 많은 편이 아닌데도 느린 부분이 있어 혹시 어디 부분에서 성능이 떨어지는지 도움 요청드립니다.

소스는 다음과 같고, 모델 클래스의 각각 Property에 DataTable의 해더명을 어트리뷰트로 정의해서 자동으로 Property의 타입에 맞게 캐스팅해서 속성을 채우는 방식입니다.

테스트시 DataTable에 약 50개 정도의 Row를 List<T>형식으로 변환 했을때 1s ~ 2s사이 정도 시간이 지연 됬습니다.

소스에 문제가 되는 부분이 있는지 도움 요청 드립니다.

public static List<T> DataTableToList<T>(DataTable table) where T : class, new()
        {
            if (table == null) return null;

            try
            {
                List<T> list = new List<T>();

                foreach (DataRow row in table.AsEnumerable())
                {
                    T obj = new T();

                    foreach (PropertyInfo prop in obj.GetType().GetProperties())
                    {
                        try
                        {
                            string headerCaption = null;

                            // 모델 클래스(T) 속성에 설정되어 있는 어트리뷰트 Get
                            object[] atts = prop.GetCustomAttributes(typeof(ExtensionEnumAttribute), false);
                            if (atts != null && atts.Length <= 0)
                            {
                                continue;
                            }

                            ExtensionEnumAttribute att = (ExtensionEnumAttribute)atts[0];
                            if (att == null || att.Objs == null || att.Objs.Length < 0 || string.IsNullOrWhiteSpace(att.Objs[0].ToString()))
                            {
                                headerCaption = prop.Name;
                            }
                            else
                            {
                                // 속성에 설정되어 있는 컬럼 헤더 타이틀 어리트뷰트 값
                                headerCaption = att.Objs[0].ToString();
                            }

                            PropertyInfo propertyInfo = obj.GetType().GetProperty(prop.Name);

                            if (propertyInfo.PropertyType.IsEnum)
                            {
                                object enumVal = Enum.Parse(propertyInfo.PropertyType, row[headerCaption].ToString());
                                propertyInfo.SetValue(obj, enumVal, null);
                            }
                            else
                            {
                                Type t = Nullable.GetUnderlyingType(propertyInfo.PropertyType) ?? propertyInfo.PropertyType;

                                if (row[headerCaption] != DBNull.Value)
                                {
                                    propertyInfo.SetValue(obj, Convert.ChangeType(row[headerCaption], t), null);
                                }
                                else
                                {
                                    propertyInfo.SetValue(obj, null, null);
                                }
                            }
                        }
                        catch(Exception ex)
                        {
                            // 데이터 테이블에 없는 모델 클래스(T) 속성
                            // 예] IsSelectCheck 속성 등..
                            continue;
                        }
                    }

                    list.Add(obj);
                }

                return list;
            }
            catch(Exception ex)
            {
                throw new Exception("데이터 테이블 -> 모델 변환 오류");
            }
        }








[최초 등록일: ]
[최종 수정일: 5/26/2017]


비밀번호

댓글 작성자
 



2017-05-26 02시32분
직접 재보시는 것이 더 빠르지 않을까요? ^^ Stopwatch 클래스를 이용해 어느 부분에서 시간이 더 걸리는 것인지 밝혀낸 다음 왜 그런지 물어보시면 더 좋을 것 같습니다.

참고로, .NET Reflection은 원래 느립니다. (느리다는 말이 매우 상대적인 기준이긴 하지만.)

아울러 다음의 글들도 한번 읽어보시고.

C# 4.0 - dynamic 키워드
; http://www.sysnet.pe.kr/2/0/766

C# - Reflection의 박싱 없이 값 형식을 다루는 방법
; http://www.sysnet.pe.kr/2/0/10866
정성태
2017-05-26 08시47분
일단 리플렉션을 메서드 사용을 최소화하는 게 중요해 보입니다.
GetProperties() 나 GetCustomAttributes() 같은 것들을 캐쉬 처리할 수 있으면 좋을 것 같습니다.
그리고 정성태님이 열거해주신 링크 중 Reflection의 박싱 없이 값 형식을 다루는 방법...에 나오는 LCG 기법이면 박싱도 해결될테구요.
어떤 방식으로 처리하는지 실제 사례를 확인해보시고 싶으시면 github의 dapper를 fork 하셔서 ormapping 하는 코드를 뒤져보시면 도움이 되지 않을까 싶습니다.
Beren Ko

... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
816박진오7/29/200910683다국어 사이트의 컨텐츠 저장 방식에 대해.. [2]
814서광원7/16/200919691IWebBrowser2를 이용한 프로그램에서 javascript의 alert 창 무시하는 법? [1]
813윤상균7/16/200910928비관리코드와의 상호운용에서 마샬링 질문 [1]
812김현우7/13/200911389usercontrol은 mdi container가 될수 없는데 이를 구현할 방법은 무엇일런지요? [2]
811조민수7/3/200910958MSDN Magazine 한글화 않되나요? [1]
810세경6/29/200916290SmartClient Vista 64bit IE7 [4]
809윤석준6/24/200916213IE -nomerge 옵션으로 새창을 열려고 합니다. [1]
808한승훈6/4/200914338dll import하기 위해 struct 구성시에 struct가 struct를 가지고 있고 포함된 struct가 ByValArray형태일때 해결 [1]
806곰티5/26/200912813defcon pro 설치 원천 봉쇄 방법 문의 [3]
802채승수5/8/200912133신뢰사이트 등록/적용에 관해 질문드립니다. [1]
801채승수4/15/200912871IE8 새세션을 코드로 구현할수 없을까요 [1]
800신동열4/7/200913309IE8에서 스마트 클라이언트 로딩 문제 [2]
7993/27/200916996이벤트 로그 오류 [1]
798천해3/26/200913274IE8.0 에 관해 질문 드립니다. [2]
797궁금..3/23/200913186IE 8 관련 질문.. [2]
796정성태3/20/200912146스마트클라이언트와 ActiveX에 관한 질문 [1]
795김기용3/19/200911759[질문] DHTML 다이얼로그 관련 [2]
794박평옥3/18/200912160Vista에서 URL Shortcut 실행 시 SetSite가 두 번 호출되는 증상에 관해 조언 부탁드립니다. [2]
792김기용3/12/200911343어제 세미나 잘 들었습니다. 질문사항이 있습니다.(ie8 마이그레이션 관련) [4]
791vb표성백2/17/200916578ATL 로 만든 COM 에 문자열 전달하기! C#에서 어떻게 하나요? [1]
790고민중2/16/20099684vista에 vs2005를 사용중입니다. [1]
789지언2/14/200911807MFC & C#(COM) 호환 관련하여 답변좀 부탁드립니다 [2]
788하루야채2/3/200911029스마트클라이언트 Windowless 설정에 대해서 문의드립니다. [2]
787궁금이2/2/200911286TFS 관련하여 질문드리고자 합니다. [2]
786맨날맑음1/30/200912364WPF를 SmartClient로 배포할순 없을까요? [2]
785정성우12/16/200811618Vista 환경에서 VB6로 개발한 어플리케이션이 IE 통해서는 런칭이 안됩니다.. [4]
... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...