.NET ListView 컨트롤의 가상모드(VirtualMode) 사용하기

.NET의 리스트뷰에는 가상모드를 지원합니다. 상당히 많은 수의 데이터를 리스트뷰의 항목으로 추가할때 속도면에서, 메모리 면에서 효율적으로 화면에 표시하기 위한 목적으로 사용할만한 매우 좋은 방법입니다.

사용자 삽입 이미지
위의 화면은 리스트뷰 컨트롤의 View 속성을 Details로 하여 그리드(Grid) 형태로 데이터를 표시하고 있습니다. 좀더 자세히 설명하면 SHP 파일에 대한 속성을 전체적으로 살펴볼 목적으로 만들어진 UI 입니다.

일반적으로 SHP 파일을 구성하는 레코드의 개수는 적게는 수십개에서 많게는 수천만개 정도됩니다. 개수가 매우 많을 경우 레코드를 한번에 리스트뷰 컨트롤에 올려 놓을 경우 UI이 얼어버리는(Freeze) 현상이 발생하고 스크롤시 매우 느린 경험을 하게 됩니다.

이럴때 가상모드를 활용해 언급한 문제를 말끔히 해결할 수 있습니다. 리스트뷰에서 가상모드를 사용하기 위해서 최소한 3가지 작업을 해줘야 합니다. 먼저 리스트뷰 컨트롤의 VirtualMode를 true로 지정하기와 다음으로 리스트뷰에서 표시할 항목의 정확한 개수를 지정하기 위한 리스트뷰 컨트롤의 VirtualListSize 값의 지정입니다. 사용자에게 제공해야할 항목의 개수가 백만개라면 이 VirtualListSize의 값을 백만으로 주면됩니다. 그리고 끝으로 실제 제공할 데이터를 필요할때 마다 그때 그때 제공해주는 RetrieveVirtualItem 이벤트 함수입니다.

private void lvAttributes_RetrieveVirtualItem(object sender, 
    RetrieveVirtualItemEventArgs e)
{
    Xr.StopMapDrawing();

    int FID = e.ItemIndex;
    XrMapLib.AttributeRow Row = SML.AttributeTable.GetRow(FID);

    if (Row != null && Row.Load())
    {
        e.Item = new ListViewItem();
        e.Item.Text = FID.ToString();
        int ToIndex = (lvAttributes.Columns.Count - 1);

        for (int iField = 0; iField < ToIndex; iField++)
        {
            String FieldValue = Row.GetValueAsString(iField);
            e.Item.SubItems.Add(FieldValue);
        }

        Row.Unload();
    }
}

위의 코드 중 중요한 부분은 e.Item에 제공할 리스트뷰의 항목을 생성해 할당한다는 것입니다. 이정도의 코드만으로도 가상모드를 충분히 실무에서 적용해 활용할 수 있지만 이에더해 캐쉬 기법을 제공함으로써 더욱 속도를 향상시킬 수 있는 방법을 제공합니다.

끝으로, 가상모드가 항상 모든 경우에서 좋은 것은 아닙니다. 필요할때마다 그때 그때 데이터를 가져오므로 전체 항목에 대한 정렬과 같은 기능을 제공할 수 없다는 점 역시 주의하시기 바랍니다.

[C#] URL을 통해 이미지(Image) 다운로드(Download)해서 파일(File)로 저장(Save)

웹상의 이미지 URL을 알고 있을때.. 해당 URL로부터 이미지에 대한 데이터를 가져와 파일로 저장하는 C# 함수입니다. 가끔 꼭 필요한 함수인데.. 필요할때 쉽게 찾아 볼 수 있도록 기록해 둡니다.

private bool DownloadRemoteImageFile(string uri, string fileName)
{
    HttpWebRequest request = (HttpWebRequest)WebRequest.Create(uri);
    HttpWebResponse response = (HttpWebResponse)request.GetResponse();
    bool bImage = response.ContentType.StartsWith("image", 
        StringComparison.OrdinalIgnoreCase);
    if ((response.StatusCode == HttpStatusCode.OK ||
        response.StatusCode == HttpStatusCode.Moved ||
        response.StatusCode == HttpStatusCode.Redirect) &&
        bImage)
    {
        using (Stream inputStream = response.GetResponseStream())
        using (Stream outputStream = File.OpenWrite(fileName))
        {
            byte[] buffer = new byte[4096];
            int bytesRead;
            do
            {
                bytesRead = inputStream.Read(buffer, 0, buffer.Length);
                outputStream.Write(buffer, 0, bytesRead);
            } while (bytesRead != 0);
        }

        return true;
    }
    else
    {
        return false;
    }
} 

사용 방법에 대한 예는 다음과 같습니다.

String url = "http://www.there.net/img.png";
String fileName = "d:/a.png";
if (!DownloadRemoteImageFile(url, fileName))
{
    MessageBox.Show("Download Failed: " + url);
}

이 요긴한 함수는 제가 작성한 것은 아니고.. 검색을 통해 찾아 테스트 해보고 실제 프로젝트에 적용해 잘… 쓰고 있는 함수입니다. 출처를 기억하지 못하지만.. 개발해 제공해 준 개발자에게 감사 드립니다.

C#은 너무 친절해…

C#은 너무 친절해… 그래서 짜증나… 므니다.. 무슨 말씀인고 허니.. C#에서 URL을 통해 웹페이지를 호출하는 코드가 있습니다.. 예를 들어서..

String urlAddress = "http://www.geoservice.co.kr:7070/wms?a|b|c";
WebRequest request = WebRequest.Create(urlAddress);

...

뭐.. 대충 위와 같은 식으로 url을 날렸더니.. url을 구성하는 문자중 |을 자동으로 %7C로 인코딩 해주십니다.. 와우~ 너무 친절하시네.. 하지만 내가 원하는 것은 그냥 | 문자를 고대로 서버로 날려야 합니다. 해서 찾아보니.. url을 문자열 그대로 날리지 않고 Uri라는 클래스를 통해 날리는 방법이 있더군요. 그래서 다음처럼 코딩했습니다.

String urlAddress = "http://www.geoservice.co.kr:7070/wms?a|b|c";
Uri uri = new Uri(urlAddress, true);
WebRequest request = WebRequest.Create(uri);

...

새롭게 추가된 2번 코드가 바로 그 녀석입니다. Uri 클래스 객체를 생성자의 두번째 인자값을 true로 주면 자동으로 |를 %7C 따위로 인코딩해주지 않아.. 제가 원하는 방법입니다. 근데 이 방식이 Deprecated 된 방식이랍니다. ㅡOㅡ;; 그럼 이와 동일한 기능을 하면서 권장하는 방식이 무엇이냐? 라고 열심히 구글링을 해봤지만 모르겠네요.. 아시는 분 계시나요?

여튼.. C#은 내부적으로 자동으로 처리해 주는 것들이 제법 있습니다. C#은 너무 친절합니다.. 그래서.. 가끔씩 맘에 않듭니다..

[C#] LINQ를 이용한 XML 파싱 쿼리

“XML에서 원하는 데이터 검색하기”에서 XML을 쿼리하는 방법을 학습한 뒤에 LINQ를 이용해 XML을 쿼리 하는 방법도 학습해 보았습니다. “XML에서 원하는 데이터 검색하기”에서 사용한 동일한 XML 데이터를 이용했고.. 그 쿼리 대상도 동일하게 했습니다. 다시 한번 쿼리 대상을 살펴보면…

  1. <book>의 개수는?
  2. 두번째 <book>의 <title>의 값은?
  3. 두번째 <book>의 genre의 값은?
  4. 두번째 <book>의 <author>의 <first-name>의 값은?

위에 대한 결과는 다음과 같습니다.

  1. 3
  2. The Confidence Man
  3. novel
  4. Herman

Linq를 이용해 위의 쿼리 대상의 결과를 얻어 내는 코드를 하나 하나 들어 보기에 앞서 LINQ를 XML에 적용하기 위한 준비 코드가 필요하며 다음과 같습니다.

XmlTextReader reader = new XmlTextReader("books.xml");
XElement xml = XElement.Load(reader);

var books = from item in xml.Descendants("book") select item;

여기서 4번 코드가 바로 LINQ입니다. 사실.. XML에서 원하는 데이터를 쿼리 하는 기능은 LINQ의 기능중 작은 부분을 차지 하고 있어 그리 크게 느껴지지 않습니다.

뭐 여튼…… 이제 우리의 목표를 하나 하나 이뤄보겠습니다. 먼저 “<book>의 개수는?”에 대한 코드입니다.

Console.WriteLine(books.Count());

다음은 “두번째 <book>의 <title>의 값은?”에 대한 코드입니다.

Console.WriteLine(
    books.ElementAt(1).Descendants("title").ElementAt(0).Value
);

그리고 세번째로 “두번째 <book>의 genre의 값은?”에 대한 코드입니다.

Console.WriteLine(books.ElementAt(1).Attribute("genre").Value);

끝으로 “두번째  <book>의 <author>의 <first-name>의 값은?”에 대한 코드입니다.

Console.WriteLine(
    books.ElementAt(1).Descendants("author").Descendants("first-name")
        .ElementAt(0).Value
);

제가 LINQ를 제대로 이해하고 있지 않은 이유일까요? LINQ를 이용해 XML에서 원하는 데이터를 쿼리하는 간결하고 직관적인 코드가 존재하지 않을까… 하는 생각이 듭니다. 바로 이런 이유가 LINQ의 이유 중에 하나이니까요… 앞서 언급했지만.. XML에서 원하는 데이터를 쿼리 하는 기능은 LINQ의 기능중 작은 부분을 차지 하고 있는듯하여 LINQ를 통해 XML을 쿼리하는 방식은 그리 매력적이라고 느껴지지 않습니다..