다음을 통해 공유


Visual Studio에서 메모리 사용량 측정(C#, Visual Basic, C++, F#)

디버거에 통합된 메모리 사용량 진단 도구를 사용하여 디버깅 중에 메모리 누수와 비효율적인 메모리를 찾아보세요. 메모리 사용량 도구는 관리되는 메모리 힙과 네이티브 메모리 힙의 하나 이상의 스냅샷을 생성하여 객체 유형의 메모리 사용량 영향에 대한 이해를 돕습니다. 디버거를 연결하지 않고 실행 중인 앱을 대상으로 지정하여 메모리 사용량을 분석할 수도 있습니다. 자세한 내용은 릴리스 또는 디버그 빌드에서 프로파일링 도구를 실행하는 방법을 참조하세요. 요구 사항에 가장 적합한 메모리 분석 도구를 선택하는 방법에 대한 자세한 내용은 메모리 분석 도구 선택을 참조하세요.

메모리 사용량 도구에서 언제든지 메모리 스냅샷을 수집할 수 있지만 Visual Studio 디버거를 사용하여 성능 문제를 조사하는 동안 애플리케이션이 실행되는 방식을 제어할 수 있습니다. 중단점, 단계별 실행, 모두 중단 및 기타 디버거 작업을 설정하면 성능 조사를 가장 관련성이 큰 코드 경로에 집중하는 데 도움이 될 수 있습니다. 앱이 실행되는 동안 이러한 작업을 수행하면 관심이 없는 코드에서 노이즈를 제거할 수 있으며 문제를 진단하는 데 걸리는 시간을 크게 줄일 수 있습니다.

중요하다

디버거 통합 진단 도구는 ASP.NET, ASP.NET Core, 네이티브/C++ 개발 및 혼합 모드(.NET 및 네이티브) 앱을 포함하여 Visual Studio에서 .NET 개발에 지원됩니다.

이 자습서에서는 다음을 수행합니다.

  • 메모리 사진 찍기
  • 메모리 사용량 현황 데이터 분석

메모리 사용량이 필요로 하는 데이터를 제공하지 않는다면, 성능 프로파일러 내의 다른 프로파일링 도구들이 여러분에게 유용할 수 있는 다양한 정보를 제공할 수 있습니다. 대부분의 경우 애플리케이션의 성능 병목 현상은 CPU, 렌더링 UI 또는 네트워크 요청 시간과 같은 메모리가 아닌 다른 항목으로 인해 발생할 수 있습니다.

메모

사용자 지정 할당자 지원 네이티브 메모리 프로파일러가 실행 시간 동안 발생되는 할당 데이터를 수집하여 ETW 이벤트 데이터를 처리합니다. CRT 및 Windows SDK의 메모리 할당자는 소스 코드 수준에서 주석 처리되어 할당 데이터를 캡처할 수 있습니다. 고유한 할당자를 작성하는 경우 myMalloc에 대한 이 예제와 같이 새로 할당된 힙 메모리에 대한 포인터를 반환하는 모든 함수를 __declspec(할당자)로 데코레이트할 수 있습니다.

__declspec(allocator) void* myMalloc(size_t size)

메모리 사용량 현황 데이터 수집

  1. Visual Studio에서 디버그하려는 프로젝트를 열고 메모리 사용량 검사를 시작할 지점에서 앱에서 중단점을 설정합니다.

    메모리 문제가 의심되는 영역이 있는 경우 메모리 문제가 발생하기 전에 첫 번째 중단점을 설정합니다.

    앱에서 메모리를 자주 할당 및 할당 해제할 때 관심 있는 작업의 메모리 프로필을 캡처하는 것은 어려울 수 있으므로 작업의 시작과 끝에 중단점을 설정하거나 작업을 단계별로 실행하여 메모리가 변경된 정확한 지점을 찾습니다.

  2. 분석하려는 코드의 함수 또는 영역 끝에 두 번째 중단점을 설정합니다(또는 의심되는 메모리 문제가 발생한 후).

  3. 진단 도구 창은 해제하지 않은 한 자동으로 표시됩니다. 창을 다시 불러오려면 디버그>Windows>진단 도구 표시을 클릭합니다.

  4. 도구 모음에서 선택 도구 설정을 사용하여 메모리 사용량을 선택합니다.

    진단 도구의 스크린샷

    진단 도구의 스크린샷

  5. 디버그 / 디버깅 시작 (또는 도구 모음의 시작, 또는 F5)를 클릭합니다.

    앱 로드가 완료되면 진단 도구의 요약 보기가 나타납니다.

    진단 도구 요약 탭의 스크린샷.

    메모

    메모리 데이터를 수집하면 네이티브 또는 혼합 모드 앱의 디버깅 성능에 영향을 줄 수 있으므로 메모리 스냅샷은 기본적으로 사용하지 않도록 설정됩니다. 네이티브 또는 혼합 모드 앱에서 스냅샷을 사용하도록 설정하려면 디버깅 세션(바로 가기 키: F5)을 시작합니다. 진단 도구 창이 나타나면 메모리 사용량 탭을 선택한 다음 힙 프로파일링선택합니다.

    스냅샷 사용 활성화 화면 캡처

    중지(바로 가기 키: Shift+F5) 디버깅을 다시 시작합니다.

    진단 도구 요약 탭의 스크린샷.

    메모

    메모리 데이터를 수집하면 네이티브 또는 혼합 모드 앱의 디버깅 성능에 영향을 줄 수 있으므로 메모리 스냅샷은 기본적으로 사용하지 않도록 설정됩니다. 네이티브 또는 혼합 모드 앱에서 스냅샷을 사용하도록 설정하려면 디버깅 세션(바로 가기 키: F5)을 시작합니다. 진단 도구 창이 나타나면 메모리 사용량 탭을 선택한 다음 힙 프로파일링선택합니다.

    스냅샷 사용 활성화 화면 캡처

    중지(바로 가기 키: Shift+F5) 디버깅을 다시 시작합니다.

  6. 디버깅 세션을 시작할 때 스냅샷을 찍으려면 메모리 사용량 요약 도구 모음에서 스냅샷 찍기를 선택합니다. (여기에서도 중단점을 설정하는 데 도움이 될 수 있습니다.)

    스냅샷 만들기 단추의 스크린샷

    스냅샷 만들기 단추의 스크린샷

    메모리 비교에 대한 기준을 만들려면 디버깅 세션이 시작될 때 스냅샷을 만드는 것이 좋습니다.

  7. 첫 번째 중단점이 적중되는 시나리오를 실행합니다.

  8. 디버거가 첫 번째 중단점에서 일시 중지되는 동안 스냅샷메모리 사용량 요약 도구 모음에서 선택하십시오.

  9. F5 눌러 두 번째 중단점에 앱을 실행합니다.

  10. 이제 다른 스냅샷을 만듭니다.

    이 시점에서 데이터 분석을 시작할 수 있습니다.

    데이터를 수집하거나 표시하는 데 문제가 있는 경우 프로파일링 오류 문제 해결 및문제 해결을 참조하세요.

메모리 사용량 현황 데이터 분석

메모리 사용량 요약 테이블의 행에는 디버깅 세션 중에 수행한 스냅샷이 나열되고 더 자세한 보기에 대한 링크가 제공됩니다.

메모리 사용량 테이블의 스크린샷

메모리 사용량 테이블의 스크린샷

열의 이름은 프로젝트 속성에서 선택한 디버깅 모드(.NET, 네이티브 또는 혼합)(.NET 및 네이티브 모두)에 따라 달라집니다.

  • 개체 (Diff) (.NET) 또는 할당 (Diff) (C++) 열에는 스냅샷이 생성될 때 .NET 또는 네이티브 메모리에 있는 개체 수가 표시됩니다.

  • 힙 크기(Diff) 열에는 .NET 및 네이티브 힙의 바이트 수가 표시됩니다.

여러 스냅샷을 생성한 경우 요약 테이블의 셀에는 행 스냅샷과 이전 스냅샷 사이의 값 변경 내용이 포함됩니다.

메모리 사용량을 분석하려면 메모리 사용량에 대한 자세한 보고서를 여는 링크 중 하나를 클릭합니다.

  • 현재 스냅샷과 이전 스냅샷 간의 차이점에 대한 세부 정보를 보려면 화살표 왼쪽의 변경 링크를 선택합니다(메모리 사용량 증가). 빨간색 화살표는 메모리 사용량이 증가했음을 나타내고 녹색 화살표는 감소를 나타냅니다.

메모리 문제를 보다 빠르게 식별하기 위해 diff 보고서는 전체 수가 가장 많이 증가하거나(개체(Diff) 열의 변경 링크를 클릭) 전체 힙 크기가 가장 많이 증가한 개체 유형별로 정렬됩니다(힙 크기(Diff) 열의 변경 링크 클릭).

  • 선택한 스냅샷의 세부 정보만 보려면 변경되지 않는 링크를 클릭합니다.

    보고서가 별도의 창에 나타납니다.

관리되는 형식 보고서

메모리 사용량 요약 테이블에서 개체(Diff) 셀의 현재 링크를 선택합니다.

메모

.NET 코드의 경우 인스턴스 보기 아이콘(개체 유형 열인스턴스 아이콘)은 디버거 통합 메모리 사용량 도구를 사용할 때 또는 힙 스냅샷을 열고 관리되는 메모리디버그를 선택하는 경우에만 사용할 수 있습니다.

위쪽 창에는 형식이 참조하는 모든 개체의 크기를 포함하여 스냅샷에 있는 형식의 수와 크기(포함 크기)가 표시됩니다.

아래쪽 창의 루트 트리에 대한 경로는 위쪽 창에서 선택한 형식을 참조하는 개체를 표시합니다. .NET 가비지 수집기는 개체를 참조하는 마지막 타입이 해제될 때에만 그 개체의 메모리를 정리합니다. 루트 트리에 대한 경로를 사용하는 방법에 대한 자세한 내용은 루트의 핫 경로 분석을 참조하세요.

매니지드 타입 리포트의 스크린샷

위쪽 창에는 형식이 참조하는 모든 개체의 크기를 포함하여 스냅샷에 있는 형식의 수와 크기(포함 크기)가 표시됩니다.

아래쪽 창의 루트 트리에 대한 경로는 위쪽 창에서 선택한 형식을 참조하는 개체를 표시합니다. .NET 가비지 수집기는 개체를 참조하는 마지막 타입이 해제될 때에만 그 개체의 메모리를 정리합니다.

참조 형식 트리는 상단 창에서 선택한 형식에 포함된 참조를 표시합니다.

참조된 개체 보고서의 스크린샷

참조 형식 트리는 상단 창에서 선택한 형식에 포함된 참조를 표시합니다.

참조된 개체 보고서의 스크린샷

위쪽 창에 선택한 형식의 인스턴스를 표시하려면 개체 유형 옆에 있는 인스턴스 보기 아이콘을 클릭합니다.

메모리 사용량 도구의 인스턴스 보기 스크린샷

인스턴스 보기는 위쪽 창의 스냅샷에 선택한 개체의 인스턴스를 표시합니다. 루트 경로 및 참조된 개체 창에는 선택한 인스턴스를 참조하는 개체와 선택한 인스턴스가 참조하는 형식이 표시됩니다. 스냅샷이 생성된 지점에서 디버거가 중지된 경우, Value 셀을 마우스로 가리키면 툴팁에 개체의 값이 표시됩니다.

메모리 사용량 도구의 인스턴스 보기 스크린샷

인스턴스 보기는 위쪽 창의 스냅샷에 선택한 개체의 인스턴스를 표시합니다. 루트 경로 및 참조된 개체 창에는 선택한 인스턴스를 참조하는 개체와 선택한 인스턴스가 참조하는 형식이 표시됩니다. 스냅샷이 생성된 지점에서 디버거가 중지된 경우, Value 셀을 마우스로 가리키면 툴팁에 개체의 값이 표시됩니다.

기본 형식 보고서

진단 도구 창의 메모리 사용량 요약 테이블에서 할당(Diff) 또는 힙 크기(Diff) 셀의 현재 링크를 선택합니다.

네이티브 형식 뷰의 스크린샷

네이티브 형식 뷰의 스크린샷

형식 뷰 스냅샷에 있는 형식의 수와 크기를 표시합니다.

  • 선택한 형식 옆에 있는 인스턴스 보기 아이콘을 선택하여 스냅샷에서 선택한 형식의 개체에 대한 정보를 표시합니다.

    인스턴스 보기에는 선택한 형식의 각 인스턴스가 표시됩니다. 인스턴스를 선택하면 할당 호출 스택 창에서 인스턴스를 생성한 호출 스택이 표시됩니다. (이 정보는 디버깅하는 동안에만 사용할 수 있습니다.)

    인스턴스 보기 및 할당 호출 스택 창의 스크린샷

  • 선택한 형식의 개체에 대한 정보를 스냅샷에서 표시하려면 해당 형식의 인스턴스 아이콘(개체 형식 열)을 선택하세요.

    인스턴스 보기에는 선택한 형식의 각 인스턴스가 표시됩니다. 인스턴스를 선택하면 할당 호출 스택 창에서 인스턴스를 생성한 호출 스택이 표시됩니다.

    인스턴스 보기 및 할당 호출 스택 창의 스크린샷

  • 스택 선택하여 선택한 형식에 대한 할당 스택을 확인합니다.

    스택 보기의 스크린샷

  • 보기 모드 목록에서 스택 보기 선택하여 선택한 형식에 대한 할당 스택을 확인합니다.

    스택 보기의 스크린샷

메모리 사용 인사이트

관리되는 메모리의 경우 메모리 분석 도구는 여러 가지 강력한 기본 제공 자동 인사이트를 제공합니다. 관리되는 형식 보고서에서 Insights 탭을 선택하면 중복 문자열, 스파스 배열이벤트 처리기 누수같은 적용 가능한 자동 인사이트가 표시됩니다.

메모리 사용량 도구의 인사이트 보기 스크린샷

중복 문자열 섹션에는 힙에 여러 번 할당되는 문자열 목록이 표시됩니다. 또한 이 섹션에서는 총 낭비된 메모리 양, 즉 (인스턴스 수 - 1) 곱하기 문자열 크기를 보여 줍니다.

스파스 배열 섹션에서는 주로 0개 요소로 채워진 배열을 보여 줍니다. 이 배열은 성능 및 메모리 사용 측면에서 비효율적일 수 있습니다. 메모리 분석 도구는 이러한 배열을 자동으로 검색하고 이러한 0 값으로 인해 얼마나 많은 메모리가 낭비되는지 보여 줍니다.

Visual Studio 2022 버전 17.9 미리 보기 1에서 사용할 수 있는 이벤트 처리기 누수 섹션에는 한 개체가 다른 개체의 이벤트를 구독할 때 발생할 수 있는 잠재적인 메모리 누수가 표시됩니다. 이벤트의 게시자가 구독자보다 더 오래 살아남는 경우, 다른 참조가 없더라도 구독자는 계속 유지됩니다. 이로 인해 메모리 누수로 이어질 수 있으며, 사용되지 않는 메모리가 제대로 해제되지 않아 애플리케이션이 시간이 지남에 따라 점점 더 많은 메모리를 사용합니다.

특정 형식에는 보유 중인 네이티브 메모리의 크기를 확인하기 위해 읽을 수 있는 필드가 있는 것으로 알려져 있습니다. Insights 탭은 개체 그래프에 가짜 네이티브 메모리 노드를 표시합니다. 이 노드는 UI가 이를 인식하고 크기 및 참조 그래프를 표시할 수 있도록 부모 개체에 의해 유지됩니다.

메모리 사용량 도구의 네이티브 인사이트 보기 스크린샷.

변경(차이) 보고서

  • 진단 도구 창에서 메모리 사용량 탭의 요약 테이블 셀에 있는 '변경' 링크를 선택하세요.

    셀의 변경 링크 선택 스크린샷

    셀의 변경 링크 선택 스크린샷

  • 관리형 보고서 또는 네이티브 보고서의 비교 목록에서 스냅샷을 선택합니다.

    비교 목록에서 스냅샷을 선택하는 스크린샷

    비교 목록에서 스냅샷을 선택한 스크린샷

변경 보고서는 기본 스냅샷 값과 비교 스냅샷의 차이를 표시하는 기본 보고서에 열((Diff)표시됨)을 추가합니다. 네이티브 타입 뷰 차이 보고서의 형태는 다음과 같습니다.

네이티브 형식 차이 보기의 스크린샷 . 네이티브 형식 차이 보기

네이티브 형식 차이 보기의 스크린샷

위쪽 창에는 형식이 참조하는 모든 개체의 크기를 포함하여 스냅샷에 있는 형식의 수와 크기(포함 크기)가 표시됩니다.

블로그 및 비디오

디버깅하는 동안 CPU 및 메모리 분석

Visual C++ 블로그: Visual C++ 2015에서의 메모리 프로파일링

다음 단계

이 자습서에서는 메모리 사용량 현황 데이터를 수집하고 분석하는 방법을 알아보았습니다. 프로파일러의 투어를 이미 완료했다면 프로파일링 도구를 사용하여 코드를 최적화하는 일반적인 방법을 알아보는 것이 좋습니다.

사례 연구: 코드 최적화에 대한 초보자 가이드

이 자습서에서는 디버깅하는 동안 메모리 사용량 현황 데이터를 수집하고 분석하는 방법을 알아보았습니다. 성능 프로파일러를 사용하여 릴리스 빌드에서 메모리 사용량을 분석하는 방법에 대해 자세히 알아볼 수 있습니다.