Microsoft MVP성태의 닷넷 이야기
.NET Framework: 684. System.Diagnostics.Process 객체의 명시적인 해제 권장 [링크 복사], [링크+제목 복사],
조회: 16888
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

System.Diagnostics.Process 객체의 명시적인 해제 권장

지난번 글에서,

System.Runtime.InteropServices.SEHException - 0x80004005
; https://www.sysnet.pe.kr/2/0/11307

Process 객체를 사용하고 일부러 CloseHandle을 통해 핸들을 닫으면 GC 상황에서 SEHException 예외가 발생한다는 것을 설명했습니다. 이를 달리 말하면, Process 객체는 GC로 인해 수행되는 Finalizer가 있다는 것이고 그 안에는 핸들을 닫는 코드가 있음을 짐작케 합니다.

실제로 Process 객체는 Component를 통해 IDisposable 인터페이스와 Finalizer를 제공받고 있으며 공통으로 호출되는 Dispose 메서드 내부에서 Close를 호출합니다. 예상할 수 있듯이 Close에서는 핸들을 닫는 코드가 있습니다.

public void Close()
{
    if (this.Associated)
    {
        if (this.haveProcessHandle)
        {
            this.StopWatchingForExit();
            this.m_processHandle.Close();
            ...[생략]...
        }
        ...[생략]...
    }
}

따라서 다음과 같은 코드일 지라도,

int pid = Process.GetCurrentProcess().Id;

많이 사용하면 Finalizer를 동반한 GC가 발생하는 것입니다. 아래는 이런 현상을 테스트할 수 있는 간단한 예제 코드입니다.

using System;
using System.Diagnostics;
using System.Threading;

namespace ConsoleApp1
{
    class Program
    {
        static void Main(string[] args)
        {
            Thread t = new Thread(outputGCFunc);
            t.IsBackground = true;
            t.Start();

            while (true)
            {
                int pid = Process.GetCurrentProcess().Id;
            }
        }

        private static void outputGCFunc()
        {
            while (true)
            {
                int count = GC.CollectionCount(2);
                Console.WriteLine($"GC 2: {count}");

                Thread.Sleep(1000);
            } 
        }
    }
}

위의 결과를 실행해 보면 GC Count가 다음과 같이 증가하는 것을 확인할 수 있습니다.

GC 2: 0
GC 2: 2
GC 2: 4
GC 2: 5
GC 2: 12
GC 2: 20
GC 2: 23
GC 2: 29
GC 2: 34
GC 2: 38
GC 2: 43
GC 2: 49
GC 2: 54
GC 2: 58
GC 2: 62
GC 2: 71

반면 while 문 내부를 이렇게 수정하면,

while (true)
{
    using (Process cur = Process.GetCurrentProcess())
    {
        int pid = cur.Id;
    }
}

GC가 0에서 거의 바뀌질 않습니다. 아울러 작업 관리자를 띄워 프로세스를 보면 메모리 관련 수치들이 거의 고정된 체로 바뀌지 않습니다. 반면, using을 사용하지 않으면 메모리가 널뛰기를 합니다.

(그렇긴 하지만, 현실적인 수준에서 문제가 될 정도로 Process 객체를 사용할 일은 없을 테니 크게 신경쓰지 않아도 되는 문제입니다. ^^)




참고로, Process.Threads 속성으로 열람해 구하는 ProcessThread 객체 역시 Component를 상속받아 Finalizer와 IDisposable 인터페이스를 물려받고 있습니다. 그렇다면, 다음과 같이 사용해야 할까요?

foreach (ProcessThread thread in cur.Threads)
{
    thread.Dispose();
}

나중에는 그래야 할지 모르겠지만, 일단 현재로써는 ProcessThread의 사용은 GC에 영향이 없으므로 Dispose의 호출 여부는 상관이 없습니다. 심지어 Component.Finalize() 조차도 영향이 없는 이유는, ProcessThread의 경우 생성자에서 GC.SuppressFinalize(this)를 호출하기 때문입니다.




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]







[최초 등록일: ]
[최종 수정일: 6/27/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 76  77  78  79  80  81  82  83  84  85  86  [87]  88  89  90  ...
NoWriterDateCnt.TitleFile(s)
11762정성태10/25/201821537사물인터넷: 52. New NodeMCU v3(ESP8266)의 http 통신파일 다운로드1
11761정성태10/25/201821458Graphics: 26. 임의 축을 기반으로 3D 벡터 회전파일 다운로드1
11760정성태10/24/201816918개발 환경 구성: 418. Azure - Runbook 내에서 또 다른 Runbook 스크립트를 실행
11759정성태10/24/201818572개발 환경 구성: 417. Azure - Runbook에서 사용할 수 있는 다양한 메서드를 위한 부가 Module 추가
11758정성태10/23/201820806.NET Framework: 800. C# - Azure REST API 사용을 위한 인증 획득 [3]파일 다운로드1
11757정성태10/19/201817284개발 환경 구성: 416. Visual Studio 2017을 이용한 아두이노 프로그램 개발(및 디버깅)
11756정성태10/19/201820531오류 유형: 500. Visual Studio Code의 아두이노 프로그램 개발 시 인텔리센스가 안 된다면?
11755정성태10/19/201821636오류 유형: 499. Visual Studio Code extension for Arduino - #include errors detected. [1]
11754정성태10/19/201818450개발 환경 구성: 415. Visual Studio Code를 이용한 아두이노 프로그램 개발 - 새 프로젝트
11753정성태10/19/201825870개발 환경 구성: 414. Visual Studio Code를 이용한 아두이노 프로그램 개발
11752정성태10/18/201818467오류 유형: 498. SQL 서버 - Database source is not a supported version of SQL Server
11751정성태10/18/201818695오류 유형: 497. Visual Studio 실행 시 그래픽이 투명해진다거나, 깨진다면?
11750정성태10/18/201816993오류 유형: 496. 비주얼 스튜디오 - One or more projects in the solution were not loaded correctly.
11749정성태10/18/201819144개발 환경 구성: 413. 비주얼 스튜디오에서 작성한 프로그램을 빌드하는 가장 쉬운 방법
11748정성태10/18/201819345개발 환경 구성: 412. Arduino IDE를 Store App으로 설치한 경우 컴파일만 되고 배포가 안 되는 문제
11747정성태10/17/201820155.NET Framework: 799. C# - DLL에도 EXE처럼 Main 메서드를 넣어 실행할 수 있도록 만드는 방법파일 다운로드1
11746정성태10/15/201819682개발 환경 구성: 411. Bitvise SSH Client의 인증서 모드에서 자동 로그인 방법파일 다운로드1
11745정성태10/15/201817450오류 유형: 495. TFS 파일/폴더 삭제 - The item [...] could not be found in your workspace, or you do not have permission to access it.
11744정성태10/15/201818366개발 환경 구성: 410. msbuild로 .pubxml 설정에 따른 배포 파일을 만드는 방법
11743정성태10/15/201819497웹: 37. Bootstrap의 dl/dt/dd 조합에서 문자열이 잘리지 않도록 CSS 설정
11742정성태10/15/201825300스크립트: 13. 윈도우 배치(Batch) 스크립트에서 날짜/시간 문자열을 구하는 방법
11741정성태10/15/201819098Phone: 13. Android - LinearLayout 간략 설명
11740정성태10/15/201821019사물인터넷: 51. Synology NAS(DS216+II)를 이용한 원격 컴퓨터의 전원 스위치 제어
11739정성태10/15/201822741Windows: 151. 윈도우 10의 전원 관리가 "균형 조정(Balanced)"으로 바뀌는 문제
11738정성태10/15/201821006오류 유형: 494. docker - 윈도우에서 실행 시 "unknown shorthand flag" 오류 [1]
11737정성태10/13/201816949오류 유형: 493. Azure Kudu - There are ... items in this directory, but maxViewItems is set to 299
... 76  77  78  79  80  81  82  83  84  85  86  [87]  88  89  90  ...