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 로 생성 후 적용하세요
[guest]

... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
1051최정수4/10/201213498WCF 클라이언트 비정상 종료 관련 [1]
1050에슈리온4/6/201215462관리자 권한과 ClickOnce, 그리고 Bootstrapper문제 [6]
1049김성혁4/4/201210177스마트클라이언트 stand-alone 방식의 배포 문제.. [1]
1048C#조으다3/27/201213915어셈블리 로드 / 언로드와 관련해서.. [6]
1047김우형3/23/201211645WCF 서비스를 이용한 데이터 전송 중 Exception 문의 [1]
1044임동찬2/29/201211500웹에서 COM Exception.. [2]
1043박성준2/24/201210322Lazy<T> 의 지연객체 생성 전에 실 객체의 Attribute분석 [2]파일 다운로드1
1042이성환2/20/201210709BlockingMethod에 빠진 스레드를 즉시 죽이고 싶습니다...;ㅅ; [2]파일 다운로드1
1039김재영2/15/20129880어셈블리에 사용자마다 다른 값을 적용하여 자동 빌드 방법이 있을까요? [2]
1037윈드로니2/12/201211356WPF 관련 질문 드립니다. [2]
1035임동찬2/9/20129703XML, XSD, XMLCodeGenerator 관련 [2]
1033곽성현2/7/20129977훔..윈도우 디바이스 드라이버 관련 [1]
1030궁빈2/2/201212378msbuild를 이용한 웹게시 및 자동 Dist 방법론에 대해 질문 드립니다!! [6]
1028신정환1/30/20129682기존 legacy Windows application이 WoA 환경에서 동작되지 않는 이유가 무얼까요? [2]
1026노현철1/19/201211440WPF가 XP에서 느리게 로딩되는 현상에 대해 질문드립니다. [4]
1025궁그미1/18/201212346네이버 소켓 로그인 이해가 안되는점이 있는데요 [2]
1022선무당1/11/201210590TFS 에서 공통 Assembly 공유 방법에 대해서 해결책이 없을까요? [2]
1021김재영1/8/20129771대리자를 메소드 파라메터로 넘겨도 됩니까? [2]파일 다운로드1
1019남산골11/13/201911888안녕하세요!! 성태님~~ [3]
1018강한구12/29/201115029WPF ClickOnce 배포시 콤포넌트 dll 다운로드 문제에 대하여 질문 있습니다. [1]
1017임동찬12/28/201112508smart client와 xbop, silverlight [6]
1015개똥이12/22/201110247RMCLOCK 설정 관련 질문이요ㅜ [2]
1011사길수12/16/201110022hyper-v 이더넷 카드 관련 문의 좀 드립니다. [1]
1006루로니12/13/201112206윈도우7과 XP의 DLL후킹 처리에 관해서 질문입니다. [3]
1005남산골12/13/201111895안녕하세요!! 첫 인사드립니다. [3]
1004뽀로로12/12/20119273Managed 어셈블리에서의 COM EntryPoint procaddress 문제 [1]
... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...