Microsoft MVP성태의 닷넷 이야기
ef core, FromSqlRaw 맵핑 질문입니다. [링크 복사], [링크+제목 복사],
조회: 2108
글쓴 사람
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,... 등의 타입을 가져갈 수밖에 없습니다.
정성태

... 16  17  18  19  20  21  22  23  24  25  26  27  [28]  29  30  ...
NoWriterDateCnt.TitleFile(s)
51837/2/20198161.NET Compact Freamwork 컨트롤러 더블버퍼링 [1]
5182wpf ...7/2/20198278wpf 질문 드립니다. [1]파일 다운로드1
51817/1/20198738DataGridview Doublebuffer 에 대해서 질문드립니다. [2]
5180김대훈6/25/20198740배열과 반복문에서 질문드립니다 [2]
5177농상6/13/20199934멀티스레드 건의 [2]
5176이선호6/13/20199517안녕하세요. 닷넷 문제로 검색하다 알게되어 들어왔습니다. 현재 IIS 문제가 있어 질문드립니다. [1]
5175Chobo6/12/20199443WPF Ellipse 그리기! [3]
5174농상6/11/20198749ThreadPool 조인에 관해 [1]
5173전우치6/9/20199271공유 리소스에 대한 스레드 동기화 처리를 위해서 lock 이용 시 문의 [3]
5172김대훈6/7/20198106너무힘드네요 공부에 대한조언부탁드립니다 [2]
5171조남석6/4/20197797EX)11-2(treeview)에 대한 질문입니다. [3]
5170레리6/4/20198409Setup 프로젝트 레지스트리 설정 관련 질문입니다. [1]파일 다운로드1
5169농상6/3/20198190멀티스레드 파라미터 관련 [2]
5168익명유저5/30/20197485항상 정말 감사드립니다... [1]
5167WPF5/23/20198389질문드립니다. [1]
5165이대희5/22/20197547Visual Studio 설치 구성요소 문의 (C# 7.3 개정판 관련) [1]
5164레드5/21/20199066실행 과정과 실행파일 디버그 시 Icon변경 질문드립니다. [5]
5163이대희5/20/20197353시작하세요 C# 7.3 프로그래밍 책 도착했습니다. [1]
5162채홍윤5/14/201910202Mono Develop window 설치 [6]
5161정대영5/13/20198076VS 2013에서 C#6.0(.netFramwork 4.6.1) $ 디버깅 오류 [1]
5160초보개발자5/13/20199077wcf 관련 국내 서적 살만한 책이 있나요? [2]
5159sdh25/9/20198627VS 2010 버전에서 생성한 DLL을 VS 2017버전에서 실행 시 에러 [2]
5158sdh5/8/20197536Visual studio 2010 버전에서 생성한 Project를 Visual studio 2017버전에서 실행에러 [1]
5157김경훈5/8/201910464Task 만들때 넘겨주는 CancellationToken은 어디서 사용 되는 건가요? [4]
5156rysoo5/8/20198717yield return의 리턴 타입 질문 드립니다. [4]
5155세퉁5/7/201910684관리자 권한으로 실행 시 알림창(?) 없이 바로 실행 시킬 수 있는 방법이 있을까요?? [2]파일 다운로드1
... 16  17  18  19  20  21  22  23  24  25  26  27  [28]  29  30  ...