베어_
TechBear
베어_
전체 방문자
오늘
어제
  • 분류 전체보기 (336)
    • Spring (33)
      • 개념 (13)
      • Security (5)
      • 실습 (1)
      • 토비 스프링 (11)
    • JPA (6)
    • 프로젝트 기록 (24)
    • DB (13)
    • JAVA (18)
    • 알고리즘 (50)
      • 유형정리 (8)
      • Baekjoon (21)
      • LeetCode (18)
    • 디자인패턴 (0)
    • 개발서적 (79)
      • Effective Java (78)
      • 객체지향의 사실과 오해 (1)
    • 독후감 (4)
    • 보안 (2)
    • 운영체제(OS) (53)
      • 공룡책 (53)
    • 컴퓨터 네트워크 (28)
      • 컴퓨터 네트워크 하향식 접근 (23)
    • 자료구조 (1)
    • DevOps (2)
    • 앱 개발 (20)
      • 안드로이드 스튜디오 (20)

블로그 메뉴

    공지사항

    인기 글

    태그

    • 데이터베이스
    • BFS
    • 알고리즘
    • C++
    • 이펙티브자바
    • 함수형인터페이스
    • 스프링
    • java
    • 코드업
    • 백준
    • 자바8
    • leetcode
    • 운영체제
    • jpa
    • 스프링시큐리티
    • 스레드
    • 토비스프링
    • Spring
    • dfs
    • 자바

    최근 댓글

    최근 글

    티스토리

    hELLO · Designed By 정상우.
    베어_

    TechBear

    개발서적/Effective Java

    [Effective Java] Serializable을 구현할지는 신중히 결정하라

    2023. 6. 6. 07:07

    어떤 클래스의 인스턴스를 직려화할 수 있게 하려면 클래스 선언에 implements Serializable만 붙이면 된다. 직렬화를 하기는 정말 쉽지만, 길게 보면 아주 값비싼 일이다.

    Seriazliable의 다양한 문제

    릴리스한 뒤에는 수정이 어렵다

    • 클래스가 Serializable을 구현하면 직렬화된 바이트 스트림 인코딩(직렬화 형태)도 하나의 공개 API가 된다.
    • 테스트할 것이 늘어난다.
    • 모든 직렬화 된 클래스에는 고유 식별 번호를 부여 받는데, 이 번호는 클래스의 모든 정보를 고려한 해싱 방법으로 생성된다. (글자 하나만 달라져도 해시값이 달라진다)

    버그와 보안에 위험이 생긴다

    • 객체는 생성자를 만드는 게 기본이다. 하지만 역직렬화는 일반 생성자의 문제가 그대로 적용되는 '숨은 생성자'이다.
    • 불변식 깨짐과 허가되지 않은 접근에 쉽게 노출된다.
      • 생성자에서 구축한 불변식을 모두 보장해야 하고 생성 도중 공격자가 객체 내부를 들여다 볼 수 없도록 해야 한다.

    Serializable의 이득과 비용을 잘 판단하자

    역사적으로 BigInteger와 Instant와 같은 '값' 클래스와 컬렉션 클래스들은 Serializable을 구현하고, 스레드 풀처럼 '동작'하는 객체를 표현하는 클래스들은 대부분 구현하지 않았다.

    상속용으로 설계된 클래스는 Serializable을 구현하지 말자

    • 인터페이스와 상속용으로 설계된 클래스는 대부분 Serializable을 구현하지 말아야 한다.

    직렬화와 확장이 모두 가능한 클래스인 경우

    • 인스턴스 필드 값 중 불변식을 보장해야 할 게 있다면 반드시 하위 클래스에서 finalize 메서드를 재정의하지 못하게 하자.
    • 인스턴스 필드 중 기본값으로 초기화되면 위배되는 불변식이 있다면 클래스에 다음의 readObjectNoData 메서드를 반드시 추가하자.
    private void readObjectNoDate() throws InvalidObjectException {
        throw new InvalidObjectException("스트림 데이터가 필요합니다.");    
    }

    Serializable을 구현하지 않을 때 주의점

    1. 상속용 클래스인데 직렬화를 지원하지 않으면 그 하위 클래스에서 직렬화를 지원하려 할 때 부담이 늘어난다.
      • 역직렬화하려면 상위 클래스는 매개변수가 없는 생성자를 제공해야 하는데, 여러분이 이런 생성자를 제공하지 않으면 하위 클래스에서 직렬화 프록시 패턴을 사용해야 한다
    2. 내부 클래스는 직렬화를 구현하지 말아야 한다.
      • 내부 클래스에는 바깥 인스턴스의 참조와 유효 범위 안의 지역변수 값들을 저장하기 위해 컴파일러가 생성한 필드들이 자동으로 추가되는데 이러한 필드들은 직렬화 형태가 분명하지 않다.
      • 단 정적 멤버 클래스는 구현해도 된다.
    저작자표시 비영리 변경금지 (새창열림)
      '개발서적/Effective Java' 카테고리의 다른 글
      • [Effective Java] readObject 메서드는 방어적으로 작성하라
      • [Effective Java] 커스텀 직렬화 형태를 고려해보라
      • [Effective Java] 공유 중인 가변 데이터는 동기화해 사용하라
      • [Effective Java] 예외를 무시하지 말라
      베어_
      베어_
      Today I learned | 문제를 해결하는 개발자

      티스토리툴바