Microsoft MVP성태의 닷넷 이야기
메모리 해제 예외 처리 관련.. [링크 복사], [링크+제목 복사]
조회: 14835
글쓴 사람
김길
홈페이지
첨부 파일
 

타인의 소스를 유지보수 해야 하는데 메모리 할당과 해제 부분의 예외 처리가 쉽지 않아 문의 드립니다.

try
{
    IntPtr hglobal = Marshal.AllocHGlobal(1);
    .............
    unsafe
    {
    fixed (~~~~~~)
    {
              .............
              hglobal = Marshal.ReAllocHGlobal(hglobal, (IntPtr)(RealSize + ReadBufSize));
              .............
    }//fixed
    }//unsafe
    .............
    Marshal.FreeHGlobal(hglobal);
    .............
}catch(~)
{
}

크게 핵심 로직을 보면 기존 개발자도 위처럼 FreeHGlobal을 호출할때
해당 포인터에 메모리가 할당되어있는지 판단하기가 어려워서 그랬는지 이게 정석인지 모르겠지만

일단 1바이트를 강제로 할당한 후에 필요한 부분에서 ReAllocHGlobal로 메모리를 재 할당하는 형태로 구현되어 있으며,
마지막에 Marshal.FreeHGlobal(hglobal);을 호출하는 형태입니다.
..... 으로 되어 있는 부분에서 상황에 따라서는 FreeHGlobal을 이용해서 메모리를 해제하거나 할당하는 로직이 있습니다.

현재 심각한 문제는 관련 메소드가 2000라인이 넘어 가는데다 메모리 누수도 있고
메모리 처리와 관련된 로직 처리가 깔끔하지 않습니다.


혹시나 해서 FreeHGlobal이 연속으로 호출되는 경우 어떤 식으로 동작하나 테스트 해보니
Marshal.FreeHGlobal(hglobal); 을 연속으로 2번 호출하게되면 어떤 경우에는 프로그램이 죽어 버리더군요.
그렇다보니 중간 중간에 예외가 발생할 경우 이미 해제된 상태에서 또 다시 해제를 시도하다 프로그램이 죽는 경우도 있습니다.

고객의 요청에 의해서 메모리 처리에 좀 더 안전을 기하기 위해
try~catch~finally 형태로 구현해서 finally부분에서 반드시 할당된 메모리를 해제하도록 하려고 하는데
로직상 예외가 아니더라도 중간 중간에 FreeHGlobal을 호출하는 경우도 있어서
finally 부분에서 hglobal에 메모리가 할당된 상태인지 아닌지를 파악하기가 어렵습니다.^^;;;



메모리가 할당된 상태인지 확인할 수 있는 방법을 몰라서
현재는 무식하지만 try문 바깥에 bool isAlloc = true; 변수 하나를 선언해서
메모리가 할당되면 true로 설정하고, 메모리를 해제했으면 false로 그때 그때 매번 값을 변경 한 뒤
finally문에서는 if (isAlloc)인 경우 Marshal.FreeHGlobal(hglobal);을 실행하고 있습니다.

이런 식으로 처리하는것 말고 좀 더 효율적인 방법이 없을런지요? ^^;;








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


비밀번호

댓글 작성자
 



2011-06-06 07시55분
엄밀히 따지면, 아무리 2000 라인이라도 isAlloc 처리가 제대로 되었다면 문제 될 상황이 아닌데 메모리 leak이 있다는 것은 다시 한번 꼼꼼하게 보셔야 할 것 같습니다.

어쨌든, 다른 방법이라면 isAlloc 플래그를 별도로 두지 마시고, Free 할 때마다 아예 hglobal = null;을 해두고 마지막에 if (hglobal == null)이라고 비교하는 것이 더 안전할 것 같습니다.

아니면, 이런 경우에 명시적으로 SafeHandle을 사용하시는 것이 좋을 것 같습니다. 다음의 문서에 보면 AllocHGlobal을 감싸는 SafeHandle을 사용하는 코드가 함께 있습니다.

SafeHandleZeroOrMinusOneIsInvalid Class
; https://learn.microsoft.com/en-us/dotnet/api/microsoft.win32.safehandles.safehandlezeroorminusoneisinvalid
정성태
2011-06-07 12시43분
[김길] 굳이 변수를 선언하지 않고 그렇게 하는 방법이 있었군요.^^

아직 unsafe나 Unmanaged Code에 대해서 제대로 이해를 못 하고 있는데
백그라운드 워커와 맞 물려있다보니 기능 하나 적용하기가 살짝 두렵네요.
알려주신 SafeHandle에 대해서 열심히 검토해봐야겠습니다.
빠른 답변 감사합니다.
[guest]

... [46]  47  48  49  50  51  52  53  54  55  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
3699박성훈2/18/201610536공부 방향성 질문 [2]
3698강준2/16/201611150Entity Framework 에서 Select for Update 가 가능한가요??? [1]
3697ds2/16/201610698어느 정도 문법을 알고 나면 [6]
3695Bere...2/15/201612949호출당한 메서드가 호출한 메서드를 알 수 있는 방법이 일반적인 방법 말고도 있을까요? [2]
3693조영준2/11/201613630UWP 앱을 만들고 있습니다. 죄송하지만 몇 가지 질문 드립니다. [4]
3692후배2/8/201612639html/css/js를 이용한 winForm의 UI표현이 가능 한가요? [3]
3690김대석2/5/201611107문의 드립니다. [2]
3687허재영2/3/201612699web api 보안관련 질문입니다. [2]
3685조진우2/1/201612637메인 응용 프로그램에서 DLL을 실시간으로 접근하여 사용하는데 이것을 모니터링 할 수 있을 까요? [1]
3684나그네1/27/201611989안녕하세요..wcf 관련하여 질문있습니다. [4]파일 다운로드1
3683김태형1/25/201612296 안녕하세요 저작도서를 구매하려는데 한가지 문의 사항이 있어서 질문드립니다. [1]
3681이영균1/8/201614861IE브라우저 추가기능관리의 항목을 사용함으로 c#코드로 변경을 하려고 합니다. [8]
3680후배1/6/201615067추천할만한 소켓 라이브러리 있으신가요? [2]
3679초록물꼬기1/6/201613940Parallel.For 에서 동기화문제에 관한 질문입니다. [11]파일 다운로드1
3678김민우1/3/201612373IHS 오류에 대한 질문입니다. [2]파일 다운로드1
3677초록색우산12/31/201513955HTTP 통신 - WebClient 이용시 한글깨짐 현상 발생 [2]
3676서정열12/31/201512805안녕하세요 WebService SoapExtensionImporter 에 대해서 질문드립니다. [3]
3675차가워12/30/201511861윈폼, 폼인폼 상황에서 하위컨트롤 포커스 문제 [1]
3673Sung...12/24/201512908UWP예외 발생 : 'System.Runtime.InteropServices.COMException'(mscorlib.ni.dll) 해결방법 [2]
3672Sung...12/23/201512161global::System.Diagnostics.Debugger.Break();가 노랗게 표시될시 [2]
3671이상준12/22/201516323한글 키보드 입력에 대해서 질문이 있습니다. [3]
3670Sung...12/18/201513092UWP 주기적으로 신호를 보내고 클라이언트에서 신호가 오면 받는 프로그램을 하려고 합니다. [5]
3669다비드12/18/201511376인터넷 임시파일 삭제 관련 질문. [1]
3668김치사발면12/16/201510840템플릿이 자꾸 초기화? 되버리네요 [1]
3667김치사발면12/15/201510751템플릿 설정 추가 질문 [1]파일 다운로드1
3666노태현12/15/201510162사용자별로 권한을 주고 볼 수 있는 데이터를 제한하려면 어떤 방법으로 접근하는게 좋을까요? [2]
... [46]  47  48  49  50  51  52  53  54  55  56  57  58  59  60  ...