Java에서 메모리 누수를 일으키는 코드를 작성하는 방법
이 텍스트는 StackOverflow Q&A 웹사이트의 인기 있는 토론인 Java에서 메모리 누수를 유발하는 코드를 작성하는 방법에서 나온 것입니다.
Q: 방금 인터뷰에 참여했는데, 면접관이 메모리 누수를 일으키는 Java 코드를 작성하는 방법을 물었습니다. 이 질문에 대해서는 잘 모르겠습니다. 너무 당황스럽습니다.
A1: 메모리 누수는 다음 단계를 통해 쉽게 발생할 수 있습니다(프로그램 코드는 특정 개체에 액세스할 수 없지만 여전히 메모리에 저장됩니다).
애플리케이션은 장기 실행 스레드(또는 스레드 풀을 사용하면 메모리가 더 빨리 누출됩니다.
스레드는 클래스 로더를 통해 클래스를 로드합니다(사용자 정의 가능).
이 클래스는 큰 메모리 블록(예: 새 바이트[1000000])을 할당하고 강력한 참조를 정적 변수에 저장한 다음 자체 참조를 ThreadLocal에 저장합니다. 추가 메모리 할당 새 바이트[1000000]는 선택 사항이지만(클래스 인스턴스 누수로 충분함) 이렇게 하면 메모리 누수가 더 빨라집니다.
스레드는 사용자 정의 클래스 또는 클래스를 로드하는 클래스 로더를 정리합니다.
위 단계를 반복하세요.
클래스 및 클래스 로더에 대한 참조가 없으므로 ThreadLocal의 저장소에 접근할 수 없습니다. ThreadLocal은 객체에 대한 참조를 보유하며 클래스 및 해당 클래스 로더에 대한 참조도 보유하므로 클래스 로더는 로드하는 클래스에 대한 모든 참조를 보유하므로 GC가 ThreadLocal에 저장된 메모리를 회수할 수 없습니다. 많은 JVM 구현에서 Java 클래스와 클래스 로더는 GC를 수행하지 않고 permgen 영역에 직접 할당되므로 더 심각한 메모리 누수가 발생합니다.
이 누수 패턴의 한 가지 변형은 어떤 형태로든 ThreadLocal을 사용하는 애플리케이션 및 애플리케이션 컨테이너(예: Tomcat)를 자주 재배포하는 경우 메모리 누수가 쉽게 발생한다는 것입니다(애플리케이션 컨테이너가 이전처럼 ThreadLocal을 사용하기 때문입니다). 설명에서는 애플리케이션이 재배포될 때마다 새로운 클래스 로더를 사용합니다.
A2:
정적 변수 참조 객체
class MemorableClass { static final ArrayList list = new ArrayList(100); }
긴 문자열의 String.intern() 호출
String str=readString(); // read lengthy string any source db,textbox/jsp etc.. // This will place the string in memory pool from which you cant remove str.intern();
닫히지 않음 열린 스트림 (파일, 네트워크 등)
try { BufferedReader br = new BufferedReader(new FileReader(inputFile)); ... ... } catch (Exception e) { e.printStacktrace(); }
닫히지 않은 연결
try { Connection conn = ConnectionFactory.getConnection(); ... ... } catch (Exception e) { e.printStacktrace(); }
JVM의 GC 도달 불가능한 영역
예를 들어 네이티브 메소드를 통해 할당된 메모리입니다.
애플리케이션 범위의 웹 애플리케이션 객체, 애플리케이션이 다시 시작되지 않았거나 명시적으로 제거되지 않음
getServletContext().setAttribute("SOME_MAP", map);
web 무효화되지 않았거나 명시적으로 제거되지 않은 세션 범위의 애플리케이션 개체
session.setAttribute("SOME_MAP", map)
올바르지 않거나 부적절한 JVM 옵션
예 , IBM JDK의 noclassgc는 쓸모 없는 클래스의 가비지 수집을 방지합니다.
A3: HashSet이 hashCode() 또는 equals()를 올바르게 구현하지 않는 경우(또는 구현하지 않는 경우) "사본"이 계속 추가됩니다. 세트. 컬렉션이 무시해야 하는 요소를 무시하지 못하면 컬렉션의 크기는 계속 커질 수 있으며 요소는 삭제할 수 없습니다.
잘못된 키-값 쌍을 생성하려는 경우 다음과 같이 할 수 있습니다.
class BadKey { // no hashCode or equals(); public final String key; public BadKey(String key) { this.key = key; } } Map map = System.getProperties(); map.put(new BadKey("key"), "value"); // Memory leak even if your threads die.
A4: 잊어버린 리스너 외에도 정적 참조, 해시맵의 키가 잘못되었습니다. /is 수명 주기를 종료할 수 없는 수정 또는 스레드 차단과 같은 일반적인 메모리 누수 시나리오는 주로 스레드와 관련된 Java의 덜 명확한 메모리 누수 상황입니다.
runtime.addShutdownHook 이후에는 Runtime.addShutdownHook이 제거되지 않습니다. RemoveShutdownHook을 사용하더라도 시작되지 않은 스레드에 대한 ThreadGroup 클래스의 버그로 인해 재활용되지 않아 ThreadGroup에서 메모리 누수가 발생할 수 있습니다.
스레드를 생성했지만 시작하지 않음, 위와 동일한 상황
ContextClassLoader 및 AccessControlContext를 상속하는 스레드를 생성하고 ThreadGroup 및 InheritedThreadLocal을 사용합니다. 이러한 모든 참조는 잠재적인 누수이며 모두 클래스가 로드됩니다. 클래스로더 및 모든 정적 참조 등에 의해. 이는 전체 j.u.c.Executor 프레임워크(java.util.concurrent)의 중요한 구성 요소인 ThreadFactory 인터페이스에 매우 분명한 영향을 미치며 많은 개발자는 잠재적인 위험을 인식하지 못했습니다. 그리고 많은 라이브러리가 요청에 따라 스레드를 시작합니다.
ThreadLocal 캐싱은 많은 경우에 좋은 방법이 아닙니다. ThreadLocal을 기반으로 하는 간단한 캐싱 구현이 많이 있지만 스레드가 예상 수명 주기를 벗어나 계속 실행되면 ContextClassLoader가 누출됩니다. 꼭 필요한 경우가 아니면 ThreadLocal 캐싱을 사용하지 마세요.
ThreadGroup.destroy()는 ThreadGroup 자체에 스레드가 없지만 여전히 하위 스레드 그룹이 있는 경우 호출됩니다. 메모리 누수가 발생하면 스레드 그룹을 상위 스레드 그룹에서 제거할 수 없으며 하위 스레드 그룹을 열거할 수 없습니다.
WeakHashMap을 사용하면 value를 직접(간접적으로) key를 참조하는데, 이는 찾기 어려운 상황이다. 이는 Weak/SoftReference를 상속하는 클래스가 보호된 개체에 대한 강력한 참조를 보유할 수 있는 경우에도 적용됩니다.
http(s) 프로토콜의 java.net.URL을 사용하여 리소스를 다운로드하세요. KeepAliveCache는 시스템 ThreadGroup에 새 스레드를 생성하여 현재 스레드의 컨텍스트 클래스 로더 메모리가 누출되도록 합니다. 스레드는 살아남은 스레드가 없을 때 첫 번째 요청에서 생성되므로 누수가 발생할 가능성이 매우 높습니다. (이 문제는 Java 7에서 수정되어 스레드 생성 코드가 컨텍스트 클래스 로더를 적절하게 제거합니다.)
인플레이터의 end()를 호출하지 않고 생성자(예: PNGImageDecoder)에 새 java.util.zip.Inflater()를 전달하려면 InflaterInputStream을 사용하세요. new만 사용하면 매우 안전하지만 생성자 매개변수로 클래스를 직접 만들고 스트림의 close()를 호출했는데 팽창기를 닫을 수 없으면 메모리 누수가 발생할 수 있습니다. 이는 종료자에 의해 해제되므로 실제로 메모리 누수는 아닙니다. 그러나 이로 인해 많은 기본 메모리가 소비되어 Linux의 oom_killer가 프로세스를 종료하게 됩니다. 따라서 이것이 우리에게 주는 교훈은 가능한 한 빨리 기본 리소스를 릴리스한다는 것입니다.
좀 더 심각한 java.util.zip.Deflater도 마찬가지입니다. 좋은 점은 디플레이터를 거의 사용하지 않는다는 점일 수 있습니다. Deflator 또는 Inflater를 직접 생성하는 경우 end()를 호출해야 한다는 점을 기억하세요.

핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

AI Hentai Generator
AI Hentai를 무료로 생성하십시오.

인기 기사

뜨거운 도구

메모장++7.3.1
사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

드림위버 CS6
시각적 웹 개발 도구

SublimeText3 Mac 버전
신 수준의 코드 편집 소프트웨어(SublimeText3)

뜨거운 주제











Java의 Weka 가이드. 여기에서는 소개, weka java 사용 방법, 플랫폼 유형 및 장점을 예제와 함께 설명합니다.

Java의 Smith Number 가이드. 여기서는 정의, Java에서 스미스 번호를 확인하는 방법에 대해 논의합니다. 코드 구현의 예.

이 기사에서는 가장 많이 묻는 Java Spring 면접 질문과 자세한 답변을 보관했습니다. 그래야 면접에 합격할 수 있습니다.

Java 8은 스트림 API를 소개하여 데이터 컬렉션을 처리하는 강력하고 표현적인 방법을 제공합니다. 그러나 스트림을 사용할 때 일반적인 질문은 다음과 같은 것입니다. 기존 루프는 조기 중단 또는 반환을 허용하지만 스트림의 Foreach 메소드는이 방법을 직접 지원하지 않습니다. 이 기사는 이유를 설명하고 스트림 처리 시스템에서 조기 종료를 구현하기위한 대체 방법을 탐색합니다. 추가 읽기 : Java Stream API 개선 스트림 foreach를 이해하십시오 Foreach 메소드는 스트림의 각 요소에서 하나의 작업을 수행하는 터미널 작동입니다. 디자인 의도입니다

Java의 TimeStamp to Date 안내. 여기서는 소개와 예제와 함께 Java에서 타임스탬프를 날짜로 변환하는 방법에 대해서도 설명합니다.

캡슐은 3 차원 기하학적 그림이며, 양쪽 끝에 실린더와 반구로 구성됩니다. 캡슐의 부피는 실린더의 부피와 양쪽 끝에 반구의 부피를 첨가하여 계산할 수 있습니다. 이 튜토리얼은 다른 방법을 사용하여 Java에서 주어진 캡슐의 부피를 계산하는 방법에 대해 논의합니다. 캡슐 볼륨 공식 캡슐 볼륨에 대한 공식은 다음과 같습니다. 캡슐 부피 = 원통형 볼륨 2 반구 볼륨 안에, R : 반구의 반경. H : 실린더의 높이 (반구 제외). 예 1 입력하다 반경 = 5 단위 높이 = 10 단위 산출 볼륨 = 1570.8 입방 단위 설명하다 공식을 사용하여 볼륨 계산 : 부피 = π × r2 × h (4

Spring Boot는 강력하고 확장 가능하며 생산 가능한 Java 응용 프로그램의 생성을 단순화하여 Java 개발에 혁명을 일으킨다. Spring Ecosystem에 내재 된 "구성에 대한 협약"접근 방식은 수동 설정, Allo를 최소화합니다.
