Microsoft MVP성태의 닷넷 이야기
글쓴 사람
김현준 (munga0828 at naver.com)
홈페이지
첨부 파일
 
안녕하세요?
Datagridview를 통해서 100만개정도의 데이터를 출력시키는 프로그램을 작성하였습니다.
프로그램 사용하는데 있어서 큰 지장은 없지만 약간의 버벅임이 존재하여 확인해본 결과 스크롤시에 GC를 매우 여러번 호출하는 현상을 발견하였습니다.
Virtual Mode를 사용하여 Cellvalueneeded 이벤트를 통해 기존에 존재하는 배열의 인덱스를 참조하여 gridview의 셀에다가 데이터를 넣기만 하는데 GC를 왜이렇게 많이 호출하는지 궁금합니다.
(제 생각에는 매모리 할당 해제가 일어나지 않을것같거든요,.,)

제가 작성한 코드는 아래와 같습니다.
여러번 확인했는데 문제될것은 없어 보이는데.. 혹시 조금 더 효율적인 방법이 있나요?

private void Form1_CellValueNeeded(object sender, DataGridViewCellValueEventArgs e)
        {
            if (e.RowIndex >= ListCF.Count)
            {
                return;
            }

            DataGridView dgv = (DataGridView)sender;

            if (dgv.SelectedCells.Count == 1)
            {
                if (e.RowIndex == dgv.SelectedCells[0].RowIndex && e.ColumnIndex == 0)
                {
                    dgv.Rows[e.RowIndex].DefaultCellStyle.BackColor = Color.LightSkyBlue;
                }
                else if (e.ColumnIndex == 0)
                {
                    dgv.Rows[e.RowIndex].DefaultCellStyle.BackColor = Color.White;
                }
            }
            else
            {
                dgv.Rows[e.RowIndex].DefaultCellStyle.BackColor = Color.White;
            }

            if (dgv.Name == DGV[(int)DGV_DEVICE.CF190].Name)
            {

                switch (e.ColumnIndex)
                {
                    case (int)DGV_HEADER_CF190.NO:
                        e.Value = e.RowIndex + 1;
                        break;
                    case (int)DGV_HEADER_CF190.NAME:
                        if (ListCF[e.RowIndex].rawData_64 == 0xF000000000000000 || ListCF[e.RowIndex].rawData_64 == 0)
                        {
                            e.Value = "-";
                        }
                        else if (DeviceDataInfo[(int)DGV_DEVICE.CF190].ContainsKey(ListCF[e.RowIndex].instructionInt))
                        {
                            e.Value = (DeviceDataInfo[(int)DGV_DEVICE.CF190][ListCF[e.RowIndex].instructionInt].InstName);
                        }
                        else
                        {
                            e.Value = "Instruction ERR";
                        }
                        break;
                    case (int)DGV_HEADER_CF190.INSTRUCTION_s:
                        e.Value = ListCF[e.RowIndex].siSnsrInst65;
                        break;
                    case (int)DGV_HEADER_CF190.BIT13__9:
                        e.Value = ListCF[e.RowIndex].siSnsrBit13__9;
                        break;
                    case (int)DGV_HEADER_CF190.PARITY_s:
                        e.Value = ListCF[e.RowIndex].siSnsrParity;
                        break;
                    case (int)DGV_HEADER_CF190.BIT7__4:
                        e.Value = ListCF[e.RowIndex].siSnsrBit7__4;
                        break;
                    case (int)DGV_HEADER_CF190.DATA_IN_s:
                        e.Value = ListCF[e.RowIndex].siSnsrData;
                        break;
                    case (int)DGV_HEADER_CF190.TFF_s:
                        e.Value = ListCF[e.RowIndex].soSnsrTFF;
                        break;
                    case (int)DGV_HEADER_CF190.TST_s:
                        e.Value = ListCF[e.RowIndex].soSnsrTST;
                        break;
                    case (int)DGV_HEADER_CF190.EOP_s:
                        e.Value = ListCF[e.RowIndex].soSnsrEOP;
                        break;
                    case (int)DGV_HEADER_CF190.SID:
                        e.Value = ListCF[e.RowIndex].soSnsrSID;
                        break;
                    case (int)DGV_HEADER_CF190.DATA_Out_s:
                        e.Value = ListCF[e.RowIndex].soSnsrOutputData;
                        break;
                    case (int)DGV_HEADER_CF190.INSTRUCTION_m:
                        e.Value = ListCF[e.RowIndex].siMdulInstruction;
                        break;
                    case (int)DGV_HEADER_CF190.PARITY_m:
                        e.Value = ListCF[e.RowIndex].siMdulParity;
                        break;
                    case (int)DGV_HEADER_CF190.DATA_IN_m:
                        e.Value = ListCF[e.RowIndex].siMdulInputData;
                        break;
                    case (int)DGV_HEADER_CF190.TFF_m:
                        e.Value = ListCF[e.RowIndex].soMdulTFF;
                        break;
                    case (int)DGV_HEADER_CF190.TST_m:
                        e.Value = ListCF[e.RowIndex].soMdulTST;
                        break;
                    case (int)DGV_HEADER_CF190.EOP_m:
                        e.Value = ListCF[e.RowIndex].soMdulEOP;
                        break;
                    case (int)DGV_HEADER_CF190.BIT12__11:
                        e.Value = ListCF[e.RowIndex].soMdulBit12__11;
                        break;
                    case (int)DGV_HEADER_CF190.WDF:
                        e.Value = ListCF[e.RowIndex].soMdulWDF;
                        break;
                    case (int)DGV_HEADER_CF190.BIT9__8:
                        e.Value = ListCF[e.RowIndex].soMdulBit9__8;
                        break;
                    case (int)DGV_HEADER_CF190.DATA_OUT_m:
                        e.Value = ListCF[e.RowIndex].soMdulData;
                        break;

                }
            }
        }






[최초 등록일: ]
[최종 수정일: 7/3/2017 ]


비밀번호

댓글 쓴 사람
 



2017-07-03 01시37분
위의 코드만 봐서는 구현상 문제는 없어 보이는데요. 스크롤할 때 Form1_CellValueNeeded 코드만 실행되는 것이 맞나요? 아래의 예제 코드로도 100만개 리스트면 재현이 되나요?

How to: Implement Virtual Mode in the Windows Forms DataGridView Control
; https://docs.microsoft.com/en-us/dotnet/framework/winforms/controls/how-to-implement-virtual-mode-in-the-windows-forms-datagridview-control
정성태
2017-09-16 08시13분
[1] 이벤트 시작 부분의 데이터그리드뷰 생성 부분에 넘어온 구리드 객체를 바로 타입변환을 하지마시고 new 로 생성 후 적용하세요
[손님]

[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
5397나그네10/15/202055.net Core 3.1 에서 Entity Framework 와 ADO.NET 선택에 관해 여쭤봅니다. [2]
5396여정욱10/15/202051CLR heap 관련 질문 2 [2]
5395여정욱10/14/202077CLR heap 관련 질문 [2]
5394진우10/12/202091닷넷코어 (닷넷5) winform wpf는 리눅스/맥에서도 가능한가요? [2]
5393김세용9/23/2020308C#에서 대량의 클래스를 빠르게 생성하는 방법이 없을까요? [6]
5392전경호9/22/2020170WPF에서 WindowsFormsHost의 메모리 누수 문제 때문에 문의드립니다. [1]파일 다운로드1
5391민성9/22/2020112안녕하세요 항상 감사드립니다. 하나 질문 드리겠습니다. [1]
5390alower9/18/2020127System.AccessViolationException 보호된메모리 부분 예외처리 [1]
5389C# 8.0 구매자9/18/2020234후위 증감 연산자 오버로딩 방법 좀 알려주세요 [4]
5388영귤9/17/2020157Nullable reference type 에 Non-nullable reference type 을 대입해도 경고가 발생하지 않습니다. [2]
5387하태9/17/2020166안녕하세요! 비동기 통신과 관련하여 질문하나만 드리겠습니다! [3]
5386박민웅9/16/2020245정성태 스승님 안녕하세요 !! [1]
5385영귤9/12/20201913항 연산자에 ref 지원? [1]
5384손님9/10/2020269시작하세요! C# 8.0 프로그래밍 책에 오타가 있는 것 같습니다. [3]
5383민성9/8/2020266안녕하세요 자주 도움을 주셔서 감사드립니다. WPF에서 크롬 브라우저 삽입에 대하여 [1]
5382정씨9/4/2020346[C#] 시리얼통신 수신된 데이터를 그리드뷰에 뿌르는데 일정 시간이 지나면 버벅 거리는 이유가 뭘까요?? [3]
5381질문 있어요9/4/2020299웹으로 사용하는 c#과 응용프로그램으로 사용하는 c#이 많이 다른가요?? [2]
5380yeon9/3/2020259winform 의 datagridview는 바인딩이 안 되나요? [1]
5379손님9/2/2020308시작하세요! C# 8.0 프로그래밍 책에 오타가 있습니다. [3]
5378net9/2/2020259mes 응용프로그램을 .net 웹폼으로 만들기도 하나요? [2]
5377net9/2/2020233.net 사용한 웹폼으로 응용프로그램을 만들경우 [1]
5376영귤9/1/2020219nullable 타입간의 비교연산은 그냥 가능한 건가요? [1]
5375영귤8/30/2020332이벤트와 델리게이트의 차이가 궁금합니다. [2]
5374손님8/27/2020343시작하세요! C# 8.0 프로그래밍 책에 오타가 있습니다. [1]
5373agj8/26/2020235System.Single::ToString()함수의 IL코드를 보았는데 ldnull, throw가 전부입니다. 왜 이렇게 짧죠? [2]
[1]  2  3  4  5  6  7  8  9  10  11  12  13  14  15  ...