static class와 static method
static
개념
어떤 객체에 소속된 것이 아닌 클래스에 고정되어있는 변수나 메서드
자바에서 static 키워드
JVM이 시작될 때 static 영역에 한번 저장되어 프로그램이 종료될 때 해제되는 것을 의미
특징
✔️ 메모리에 고정적으로 할당 됨
각 객체들에서 공통적으로 하나의 값이 유지되어야할 경우 유용하게 사용
✔️ 객체 생성 없이도 메서드나 변수 사용 가능
자동으로 메모리의 static영역에 올라가기 때문에 객체 생성할 필요없이 사용 가능
✔️ static 영역 적재
프로그램의 시작과 끝까지 메모리에 존재
프로그램 시작 시, 메모리의 static영역에 적재
프로그램 종료 시, 메모리의 static영역에서 해제
무분별한 static 남발 조심
과도하게 많은 static을 선언할 경우 메모리에 과부하
시작 시, static 영역의 메모리 크기를 넘어서는 양의 static 메서드들이 존재시 에러 발생 가능
staic영역에 할당된 메모리는 모든 객체가 공유하는 메모리
일반적인 메서드는 객체 생성 시 메모리의 Heap영역에 올라감. Heap영역은 GC에 의해 자동으로 메모리가 관리되나, static영역은 해당 관리를 받지 못함
✔️ static매서드 내에서는 인스턴스 변수 사용 불가
인스턴스 변수
런타임 시점에 메모리에 올라감
객체를 생성해야 사용 가능
Static 메서드
컴파일 시점에 메모리에 올라감
메모리 효율
긍정적인 메모리 효율 측면
객체를 생성하지 않고 사용 가능
GC가 대상 객체로 생각하지 않아 성능향상에 도움을 줄 수 있음
부정적인 메모리 효율 측면
GC의 대상이 아님
static 할당 값이 클 경우 GC의 효율이 떨어짐
과도한 static의 경우 메모리 누수 발생 가능
static영역의 특징
프로그램이 끝날 때까지 메모리에서 내릴 수 없음
static영역의 차지 영역이 클 경우 메모리 부족 발생 가능
문제점
전체 프로그램과 동일한 라이프 사이클
static멤버는 사용유무와 상관없이 프로그램 시작과 끝까지 메모리 내 존재
static 변수의 생성과 소멸 지시 불가
개발자가 프로그램적으로 생성과 소멸에 관여 불가
생성
: 프로그램 로딩 시소멸
: 프로그램 종료 혹은 JVM이 내려갈 때
Thread safe 하지 않음 (
동시성 문제
)모든 스레드에서 static필드를 공유하며 프로그램 전역 사용
한 스레드에서 값 변경 시, 다른 모든 스레드에서 영향을 받음
Thread safe를 위한 추가작업 필요
static 메서드는 오버라이딩 불가
인스턴스 메소드
런타임 시 해당 메소드를 구현하고 있는 실제 객체를 찾아 호출 (
다형성 적용
)
static 메소드
JVM과 컴파일러 모두 실제 객체를 찾는 작업을 시행하지 않음
컴파일 시점에 선언된 타입의 메서드 호출 (
다형성 적용X
)
static 멤버는 Serialization 불가
객체 직렬화는 인스턴스에 대해 적용 됨
클래스 자체 정보인 static 멤버는 포함되지 않음
메모리 문제
프로그램이 종료될 때까지 Garbage Collector로 회수되지 않음
메모리 누수 문제 발생 가능
테스트의 어려움
프로그램 전체에서 접근할 수 있는 필드를 추론하기 어려움
static 메서드
객체 생성 없이 사용할 수 있는 메서드
빠른 속도와 공유에 효율적
사용 예시
실행 결과
Static 메서드 내에서는 인스턴스 변수 사용 불가
Counter
클래스의 getCount
메서드는 static 메서드이다. 때문에 객체 생성 없이 클래스를 통해 메서드를 직접 호출할 수 있다. 이때 주의할 점은 static 메서드 안에서는 static 변수만 접근이 가능하다.
싱글톤 패턴과 Static 매서드
static 메서드는 디자인 패턴 중 싱글톤 패턴에서 사용한다.
싱글톤 패턴
: 객체를 한번만 생성하고 그 객체를 공유하는 패턴
생성자를 private로 만든다면 외부에서 new 키워드로 생성이 불가하다. 하지만 static 메서드를 사용한다면 객체 생성 없이 접근할 수 있기 때문에 싱글톤 패턴에서 인스턴스를 가져올 때 사용한다.
예제 코드
그럼 static 메서드는 장점만 있을까? 아니다, 객체지향에서 멀어지고, 메모리 효율이 오히려 떨어질 수 있다.
객체지향
객체지향
객체들이 서로 관계를 맺고 메시지를 통해 정보를 교환하고 결과를 반환하길 기대
static 메소드의 실행
객체에게 행위를 지시하지 않음
다른 객체와 관계를 맺고 있지도 않음
메시지 전달이 아닌, (절차지향의) 함수 호출에 불과하다고 할 수 있음
다형성의 위반
컴파일 시 정적 바인딩 되기 때문에, static 메서드는 오버라이딩과 동적 바인딩이 불가
static 클래스
클래스에 static을 붙일 땐
내부 클래스
에 붙이는 경우가 많음외부 클래스의 인스턴스 생성없이 내부 클래스를 접근하기 위한 용도
Inner 클래스
: 클래스 내부에 있는 클래스
Inner 클래스의 문제점
외부 참조를 한다는 점
메모리 누수 현상이 발생한다는 점
외부 참조
Inner 클래스를 생성하면 내부 클래스가 외부의 멤버를 사용하지 않아도, 숨겨진 외부 참조가 생성된다.
즉 비정적 멤버 클래스의 인스턴스는 외부 클래스의 인스턴스와 암묵적으로 연결된다.
예시 코드
위의 코드는 Outer_Class.class
파일이 생성된다. 그리고 Outer_Class$Inner_Class.class
파일도 같이 생성된다. 이 class 파일을 바이트 코드로 디컴파일 시켜보면 아래와 같이 외부 클래스를 매개변수로 받아 인스턴스 변수로 저장하는 것을 확인 할 수 있다.
이때 this$0
는 OuterClass를 참조하는 있는 바이트 코드에서만 보여지는 Hidden 변수이다.
메모리 누수
Inner 클래스에서 외부 클래스는 필요가 없어지고 내부 클래스만 남아있을 경우, 외부 클래스가 GC 대상이 되어 바로 사라져야 할거 같지만, 외부 참조로 인해 내부 클래스와 연결되어 있어 사라지지 않는다. 메모리에서 제거가 안되고 이는 메모리 누수로 연결되어 프로그램이 터진다.
메모리 누수: 컴퓨터 프로그램이 필요하지 않은 메모리를 계속 점유하고 있는 현상
예시 코드로 살펴보자.
Outer_Class
위의 코드의 실행과정을 따라가보자.
Inner_Class 를 담을 리스트 arr 생성
반복문을 통해 50개의 Inner_Class 객체를 리스트의 담는다.
이때 내부 클래스 생성을 위해 외부 클래스를 먼저 인스턴스화 해야하기에, Outer_Class를 new 키워드로 인스턴스화한다..
Outer_Class 생성자 파라미터로 1억 사이즈의 int 배열이 생성되어 400MB 크기의 객체가 힙 메모리에 쌓인다.
그리고
getInnerObject()
메서드를 호출하여 Inner_Class 객체를 생성하고 반환한다.이제 생성된 Inner_Class 객체를 리스트에 저장한다.
메서드 호출용도로 일회용으로 사용된 400MB의 Outer_Class 객체는 더이상 필요없게 되어 GC 대상이 되어 힙 메모리에서 삭제가 되어야 할 것이다?
위 코드를 실행해보면 얼마 못가 OutOfMemoryError 예외가 발생한다. GC가 정상적으로 Unreachable 데이터를 수거하지 못해 메모리 관리가 불가하여 메모리 누수가 됐기 때문이다. 메서드 호출만을 위한 일회용 객체가 GC 수거 대상이 되지 못하여 400MB 데이터가 계속 쌓이게 되었고, 프로그램이 터진 것이다.
Inner 클래스는 static으로 선언하자
앞서 설명한 문제점들을 해결할 수 있는 것이 static class 이다.
static 클래스로 만든 Inner 클래스는 외부 인스턴스 없이 사용하기에 외부 참조를 하지 않는다. 때문에 일회용으로 사용된 외부 클래스 객체는 더이상 내부 클래스와 관계가 없기 때문에 GC 수거 대상이 되어 메모리 관리가 가능하다.
static 클래스가 아닌 클래스는 얼마 못가 OutOfMemoryError 를 던졌지만 static Inner class 는 끝까지 프로그램이 수행되었다.
static 잘 알고 사용하자.
ref
정적 메소드, 너 써도 될까? 07-03 스태틱 ☕ 내부 클래스는 static 으로 선언 안하면 큰일 난다 [Java] static변수와 static 메소드 Static 사용을 피해야 하는 이유 [JAVA] 정적 메소드를 오버라이드 하지 못하는 이유
Last updated