Microsoft MVP성태의 닷넷 이야기
ef core, FromSqlRaw 맵핑 질문입니다. [링크 복사], [링크+제목 복사],
조회: 10207
글쓴 사람
victor (jys923 at gmail.com)
홈페이지
첨부 파일
 

.FromSqlRaw 의 결과가 동적입니다.

sn : result1 : result2 : result3 : result4 : result5 :

result1 결과가 몇개가 될지 모릅니다.

그래서 맵핑하는 클래스에 string sn과 List<int> result 를 만들었는데 맵핑이 잘안되네요.


int numberOfResults = 5; // 생성할 서브쿼리 개수

StringBuilder sqlBuilder = new StringBuilder();
for (int i = 1; i <= numberOfResults; i++)
{
    sqlBuilder.AppendLine($"(SELECT Result FROM Inspects WHERE ProbeSNId = main.ProbeSNId AND InspectTypeId = {i} AND DataFlag = 1) AS {i},");
}

// 마지막 줄에는 쉼표를 제거
sqlBuilder.Remove(sqlBuilder.Length - 3, 1);

string sql = @"
            SELECT CONCAT(pt.Code, pst.DateTime, LPAD(pst.PcNo, 3, '0'), LPAD(ps.ProbeSeqNo,5,'0')) AS ProbeSN,
                   {sqlBuilder}
            FROM (SELECT DISTINCT ProbeSNId FROM Inspects) AS main
            JOIN ProbeSNs ps ON main.ProbeSNId = ps.Id
            JOIN ProbeSNTypes pst ON ps.ProbeSNTypeId = pst.Id
            JOIN ProbeTypes pt ON ps.ProbeTypeId = pt.Id";

List<ProbeSNView> probeSNViews = db.ProbeSNViews
    .FromSqlRaw(sql)
    .AsEnumerable()
    .Select(x => new ProbeSNView
    {
        ProbeSN = x.ProbeSN,
        Result = GetDynamicResults(x).ToList(),
    })
    .ToList();

private static List<int> GetDynamicResults(dynamic x)
{
    var results = new List<int>();

    for (int i = 1; i <= 5; i++)
    {
        var result = x[i] as int?;

        if (result.HasValue)
        {
            results.Add(result.Value);
        }
    }

    return results;
}


//[NotMapped]
public class ProbeSNView
{
    //[Key]
    public required string ProbeSN { get; set; }

    public IEnumerable<int>? Result { get; set; }
}

이렇게 해봤습니다.

어떻게 맵핑 하면 좋을까요?

https://github.com/jys923/ProcessAutomation.git








[최초 등록일: ]
[최종 수정일: 2/22/2024]


비밀번호

댓글 작성자
 



2024-02-22 04시29분
일단, 위의 방식대로 하면 예외(Exception) 없이 잘 동작했나요? 만약 예외 없이 코드가 잘 수행된다면, 그 코드까지 테스트한 것을 github에 체크인해 주세요. 그리고 디버그 상태에서 GetDynamicResults 메서드에 BP를 걸고 x의 값을 Watch 창으로 한번 확인해 보세요.
정성태
2024-02-23 08시35분
답변 갑사합니다.
예외(Exception)가 났습니다.

그래서 방식을 바꿨습니다.

바꾼 방식 github에 체크인 했습니다.

#elif SELECT_SQL_DYNAMIC
            #region select sql dynamic 18s
            // SQL 쿼리 문자열을 생성
            StringBuilder sqlBuilder = new StringBuilder();
            for (int i = 1; i <= numberOfResults; i++)
            {
                sqlBuilder.AppendLine($"(SELECT Result FROM Inspects WHERE ProbeSNId = main.ProbeSNId AND InspectTypeId = {i} AND DataFlag = 1) AS Result{i},");
            }
            // 마지막 줄에는 쉼표를 제거
            sqlBuilder.Remove(sqlBuilder.Length - 3, 1);

            string sql = $@"
            SELECT CONCAT(pt.Code, pst.DateTime, LPAD(pst.PcNo, 3, '0'), LPAD(ps.ProbeSeqNo,5,'0')) AS ProbeSN,
                   {sqlBuilder}
                FROM (SELECT DISTINCT ProbeSNId FROM Inspects) AS main
                JOIN ProbeSNs ps ON main.ProbeSNId = ps.Id
                JOIN ProbeSNTypes pst ON ps.ProbeSNTypeId = pst.Id
                JOIN ProbeTypes pt ON ps.ProbeTypeId = pt.Id";

            List<ProbeSNView> probeSNViews = new List<ProbeSNView>();

            // 데이터베이스 연결 및 쿼리 실행
            using (var command = db.Database.GetDbConnection().CreateCommand())
            {
                command.CommandText = sql;
                db.Database.OpenConnection();
                using (var result = command.ExecuteReader())
                {
                    while (result.Read())
                    {
                        ProbeSNView probeSNView = new ProbeSNView
                        {
                            ProbeSN = result.GetString(0), // 첫 번째 컬럼은 ProbeSN
                            Result = new List<int>()
                        };
                        for (int i = 1; i <= numberOfResults; i++)
                        {
                            // Result1, Result2, ... 컬럼은 1부터 시작
                            int? value = result.IsDBNull(i) ? null : (int?)result.GetInt32(i);
                            probeSNView.Result.Add(value ?? 0); // null일 경우 0으로 대체
                        }
                        probeSNViews.Add(probeSNView);
                    }
                }
            }

            /*foreach (var item in probeSNViews)
            {
                Console.Write(item.ProbeSN + " ");
                foreach (var result in item.Result)
                {
                    Console.Write(result + ": ");
                }
                Console.WriteLine("");
            }*/
            Console.WriteLine("probeSNViews.Count : " + probeSNViews.Count());
            #endregion
#endif

그런데 기존에 데이터 형이 정해 졌던 걸로 바로 맵핑 했을떄는 10만건 기준 1초였는데 8초가 걸리네요.

성능을 개선 시킬 좋은 방법 없을까요?
victor
2024-02-23 08시48분
SELECT_LINQ

SELECT_SQL

같은 쿼리인데 linq와 sql 속도차이가 20초 정도 납니다.

linq가 느립니다.

ms에선 linq를 추천 하는것 같은데 성능차가 너무 커서요

제가 뭔가 잘못한것 같아습니다. 검토 부탁드려도 될까요?
victor
2024-02-23 11시53분
예제에서 SELECT_SQL도 내부적으로는 SqlCommand/SqlDataRader를 사용할 뿐이어서 기본적인 처리는 SELECT_SQL_DYNAMIC과 같습니다.

단지, 2개의 예제가 다른 것은 SELECT_SQL의 경우 필드가 정해져 있는 반면, SELECT_SQL_DYNAMIC은 가변적이라는 정도인데요, 일단 테스트를 위해 SELECT_SQL_DYNAMIC 코드도 다음과 같은 식으로 바꿔보세요.

while (result.Read())
{
    ProbeView probeView = new ProbeView
    {
        ProbeSN = result.GetString(0),
        Result1 = result.GetInt32(1),
        Result2 = result.GetInt32(2),
        Result3 = result.GetInt32(3),
        Result4 = result.GetInt32(4),
        Result5 = result.GetInt32(5),
    };
    probeViews.Add(probeView);
}

만약 저 차이라면 방법이 없습니다.

혹은, 만약 가변 필드가 몇 개 안 된다면 고정 타입으로 ProbeViewResult1, ProbeViewResult2, ProbeViewResult3,... 등의 타입을 가져갈 수밖에 없습니다.
정성태

... 31  32  33  34  35  [36]  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
4993학생6/3/201814497질문 드립니다 [1]
4991최진안5/24/201814829Credential Provider 질문 [2]
4990heyg...5/21/201815282Sybase Adaptive Server Anywhere 6 버전에 대한 질문입니다. [9]
4989강한음5/18/201818151clickonce 배포 후 실행 무응답 [4]
4988C#7....5/17/201814114dynamic 키워드를 사용한 객체 핸들링에 대해서 여쭤보려고 합니다. [1]
4987포플러5/14/201815748닥터왓슨 로그 - c0000005 (액세스 위반) 분석 부탁드려도 될까요? [2]
4986익명5/14/201817701비주얼 스튜디오 wpf 프로젝트에서 어떻게 하면 exe파일과 실행에 필요한 파일들을 분리해서 정리해서 디렉토리로 묶을 수 있을까요? [4]
4985대구개발자5/7/201815220새로운 폴더만 "이름 없는 파일" 오류 [1]
4984안중언5/6/201816183교재 143page [1]
4983익명5/4/201826225(wpf) 다른 컴퓨터에서 사용하면 자꾸 ('CefSharp.Core.dll' 또는 여기에 종속되어 있는 파일이나 어셈블리 중 하나를 로드할 수 없습니다)라고 떠요.ㅠㅠ [5]
4982Soul...4/27/201816367MFC ActiveX 컨트롤 안에 있는 C# ActiveX 컨트롤 포인터 얻기 [4]
4981대전박4/25/201815098WPF IValueConverter 를 구현해서 StaticResource로 사용할때요 [1]
4980대전박4/23/201816466WPF OS버전 따라 Style 적용이 안되는 프로퍼티가 있을수 있나요? [2]
4979초보개발자4/18/201821348C# 프레임워크 버전이 다른 DLL끼리의 사용 [7]파일 다운로드1
4977Soul...4/17/201815905WebBrowser 컨트롤 Script 통신 문제 [3]
4976맹가이버4/14/201816858윈도우 서비스 프로그램에서 응용프로그램 호출하는 법 [1]
4975lemo...4/11/201817389안녕하세요 네이버로그인관련 질문드립니다. [2]
4973홍길동4/6/201815221ebook 출간 계획은 없으신가요? [2]
4978홍길동4/17/201814956    답변글 [답변]: ebook 출간 계획은 없으신가요?
4972dwkim4/3/201816057EasyHook 관련 질문 [4]
4968최홍준3/30/201814808Windows 7 Credential Provider Android와 연동 [1]
4967이대희3/30/201815800비주얼 스튜디오 설치 워크로드 중에 ".NET Core 플랫폼 간 개발" 이건 뭐하는 것인지요. [1]
4965이대희3/30/201815326자마린 설치후 안드로이드 프로젝트 생성시 디자이너가 없다는 에러가 발생합니다. [3]
4969이대희3/31/201815829    답변글 [답변]: 자마린 설치후 안드로이드 프로젝트 생성시 디자이너가 없다는 에러가 발생합니다.파일 다운로드2
4970이대희4/1/201815321        답변글 [답변]: [답변]: 자마린 설치후 안드로이드 프로젝트 생성시 디자이너가 없다는 에러가 발생합니다. [1]
4963이대희3/29/201815675UWP 스터디를 위해 찰스페졸드 저자의 Programming Windows 6판은 어떠한지요? [1]
... 31  32  33  34  35  [36]  37  38  39  40  41  42  43  44  45  ...