Microsoft MVP성태의 닷넷 이야기
Java: 18. Java의 Memory Mapped File 자원 반환이 안 되는 문제 [링크 복사], [링크+제목 복사],
조회: 23480
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

Java의 Memory Mapped File 자원 반환이 안 되는 문제

Java에서는 다음의 글에 소개된 대로,

Memory Mapped File
; http://www.javacodex.com/Files/Memory-Mapped-File

이렇게 MMF를 사용한다고 합니다.

import java.io.*;
import java.nio.*;
import java.nio.channels.*;

public class MemoryMappedFile {

   public static void main(String[] args) {

      try {

         File file = new File("/tmp/myfile.txt");
         FileChannel fc = new RandomAccessFile(file, "rw").getChannel();
         ByteBuffer bb = fc.map(FileChannel.MapMode.READ_ONLY, 0, (int) fc.size());

         // print contents of ByteBuffer //
         while (bb.hasRemaining()) {
            System.out.print((char) bb.get());
         }

      } catch (IOException e) {
         System.out.println("Error: " + e.toString());
      }
   }
}

그런데, 사내 자바 개발자가 메모리 맵 파일이 GC가 발생하기 전까지는 어떻게 해도 해제할 수 없다는 문제를 경험했습니다. 이를, 간단하게 다음의 코드로 재현해 볼 수 있습니다.

package com.company;

import java.io.*;
import java.nio.*;
import java.nio.channels.*;

public class Main {

    public static void main(String[] args) {

        try {
            while (true) {

                UseMemoryMap();
                if (System.in.available() != 0) {
                    break;
                }

                Thread.sleep(1000);
            }
        } catch (IOException e) {
        } catch (InterruptedException e) {
        }
    }

    public static void UseMemoryMap() {
        try {
            File file = new File("c:/temp/Installer.exe"); // 1MB짜리 파일
            FileChannel fc = new RandomAccessFile(file, "rw").getChannel();
            MappedByteBuffer bb = fc.map(FileChannel.MapMode.READ_ONLY, 0, (int) fc.size());

            bb.clear();
            fc.close(); // clear / close를 해도 메모리는 지속적으로 증가!
        } catch (IOException e) {
            System.out.println("Error: " + e.toString());
        }
    }
}

위의 자바 코드는 1MB 파일에 대한 MMF 설정을 1초마다 반복하는 바람에 "Private Bytes"가 끊임없이 증가하는 문제가 있습니다. 실제로 위의 프로그램이 실행되는 동안 "Process Explorer"로 File Handle의 상황을 모니터링해 보면 다음과 같이 해제되지 않고 끊임없이 증가하는 것을 볼 수 있습니다.

java_mem_map_leak_1.png

이 문제를 해결하려면 명시적으로 System.gc()를 호출해야 하는데, 가령 다음과 같이 위의 코드를 수정해 주면,

try {
    while (true) {

        UseMemoryMap();
        if (System.in.available() != 0) {
            break;
        }

        Thread.sleep(1000);
        System.gc();
    }
} catch (IOException e) {
} catch (InterruptedException e) {
}

메모리 증가 문제가 없어집니다.

이 문제가 발생하는 원인은 2가지가 복합적으로 작용해 발생합니다. 하나는, Java 측에서 MMF 자원을 명시적으로 해제하는 방법을 제공하지 않는다는 점입니다. 또 하나는, Java의 GC 입장에서 GC Heap이 아닌 Native 메모리에 대한 증가는 System.gc 호출을 발생시키지 않는다는 점입니다.

이로 인해, 특정 자바 프로그램이 지속적으로 MMF를 열고/닫는 작업을 하지만 딱히 GC Heap에 대한 변동이 없는 상황이라면 저런 식으로 메모리가 무한정 소비되는 문제가 발생하는 것입니다.




다음은 사내 개발자가 제시한 부가적인 해결 방법입니다.

How to unmap a file from memory mapped using FileChannel in java?
; http://stackoverflow.com/questions/2972986/how-to-unmap-a-file-from-memory-mapped-using-filechannel-in-java

덧글에 보면, SUN JDK에서의 우회 방법이 나옵니다. 또한, 주기적으로 System.gc를 하라는 것도 나오는데 이는 "-XX:+DisableExplicitGC" 옵션이 있는 경우 동작하지 않는다고 합니다. 따라서, (호스트 환경을 장담할 수 없는) 라이브러리 개발자가 써먹기에는 부담스럽습니다.




그렇다면 혹시 닷넷은 어떨까요? 다행히, 닷넷은 MemoryMappedFile.Dispose 메서드가 제공되는데 이를 호출하면 Native 리소스가 잘 해제됩니다.

아래는 참고 글입니다.

Memory-mapped File은 Private Bytes 크기에 포함될까요?
; https://www.sysnet.pe.kr/2/0/11159




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







[최초 등록일: ]
[최종 수정일: 9/28/2024]

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

비밀번호

댓글 작성자
 




... 121  122  123  124  125  126  127  128  129  [130]  131  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1806정성태11/10/201425134.NET Framework: 477. SeCreateGlobalPrivilege 특권과 WCF NamedPipe
1805정성태11/5/201421984.NET Framework: 476. Visual Studio에서 Mono용 Profiler 개발 [3]파일 다운로드1
1804정성태11/5/201428236.NET Framework: 475. ETW(Event Tracing for Windows)를 C#에서 사용하는 방법 [9]파일 다운로드1
1803정성태11/4/201420339오류 유형: 261. Windows Server Backup 오류 - Error in backup of E:\$Extend\$RmMetadata\$TxfLog
1802정성태11/4/201422255오류 유형: 260. 이벤트 로그 - Windows Error Reporting / AEAPPINVW8
1801정성태11/4/201427601오류 유형: 259. 이벤트 로그 - Windows Error Reporting / IPX Assertion / KorIME.exe [1]
1800정성태11/4/201418254오류 유형: 258. 이벤트 로그 - Starting a SMART disk polling operation in Automatic mode.
1799정성태11/4/201423070오류 유형: 257. 이벤트 로그 - The WMI Performance Adapter service entered the stopped state.
1798정성태11/4/201431860오류 유형: 256. 이벤트 로그 - The WinHTTP Web Proxy Auto-Discovery Service service entered the stopped state. [1]
1797정성태11/4/201417570오류 유형: 255. 이벤트 로그 - The Adobe Flash Player Update Service service entered the stopped state.
1796정성태10/30/201424538개발 환경 구성: 249. Visual Studio 2013에서 Mono 컴파일하는 방법
1795정성태10/29/201427065개발 환경 구성: 248. Lync 2013 서버 설치 방법
1794정성태10/29/201422501개발 환경 구성: 247. "Microsoft Office 365 Enterprise E3" 서비스에 대한 간략 소개
1793정성태10/27/201423150.NET Framework: 474. C# - chromiumembedded 사용 - 두 번째 이야기 [2]파일 다운로드1
1792정성태10/27/201423285.NET Framework: 473. WebClient 객체에 쿠키(Cookie)를 사용하는 방법
1791정성태10/22/201423005VC++: 83. G++ - 템플릿 클래스의 iterator 코드 사용에서 발생하는 컴파일 오류 [5]
1790정성태10/22/201418535오류 유형: 254. NETLOGON Service is paused on [... AD Server...]
1789정성태10/22/201421210오류 유형: 253. 이벤트 로그 - The client-side extension could not remove user policy settings for '...'
1788정성태10/22/201423237VC++: 82. COM 프로그래밍에서 HRESULT 타입의 S_FALSE는 실패일까요? 성공일까요? [2]
1787정성태10/22/201431404오류 유형: 252. COM 개체 등록시 0x8002801C 오류가 발생한다면?
1786정성태10/22/201432718디버깅 기술: 65. 프로세스 비정상 종료 시 "Debug Diagnostic Tool"를 이용해 덤프를 남기는 방법 [3]파일 다운로드1
1785정성태10/22/201421940오류 유형: 251. 이벤트 로그 - Load control template file /_controltemplates/TaxonomyPicker.ascx failed [1]
1784정성태10/22/201430027.NET Framework: 472. C/C++과 C# 사이의 메모리 할당/해제 방법파일 다운로드1
1783정성태10/21/201423487VC++: 81. 프로그래밍에서 borrowing의 개념
1782정성태10/21/201420214오류 유형: 250. 이벤트 로그 - Application Server job failed for service instance Microsoft.Office.Server.Search.Administration.SearchServiceInstance
1781정성태10/21/201420658디버깅 기술: 64. new/delete의 짝이 맞는 경우에도 메모리 누수가 발생한다면?
... 121  122  123  124  125  126  127  128  129  [130]  131  132  133  134  135  ...