JAVA 일반 Serializable 객체직렬화

2015.10.14 12:53

졸리운_곰 조회 수:265

 

 

Serializable 객체직렬화

java 2013/07/30 17:13

기존에 개발되어 있는 소스 코드를 보다 보니 implements java.io.Serializable과 같이 Serializable 인터페이스를 구현하는 VO(Value Object) 클래스들이 많이 보인다.

혹시 RMI(Remote Method Invocation) 통신이나 네트웍 통신을 할 때 직렬화를 사용하여 연동을 하나? 싶었지만 아니였다.

원인을 알고 싶었지만 개발된지 이미 5년이 넘은 시스템이고, 개발자는 5대째 계승되고 있는 시점이다.

결국 Serializable interface가 왜 구현되어 있는지는 모르겠지만 아마 습관적으로 붙이지 않았나 싶다.

 

이런 코드를 보니 나 조차도 Serializable에 대해 자세히 알고 있나라는 생각을 가지게 되었고, 한번쯤은 정리를 해봄으로써 확실히 알아가는 것이 좋을 것 같다.

 

직렬화란?

JVM 힙 영역에 존재하는 객체를 한 줄로 늘어선 바이트의 형태로 만드는 것을 객체의 직렬화라고 하고, 객체의 형태로 복원하는 작업을 역직렬화라고 한다.

 

java.io.Serializabe 인터페이스를 보면 구현해야 하는 메소드가 없다. 그 이유는 Serializable 인터페이스를 구현한 구현체가 직렬화 대상이다라는 것을 JVM에게 알려주는 역할만을 하기 때문이다.

 

다음은 Serializable, Externalizable 인터페이스를 이용한 샘플 코드이다.

Serializable 인터페이스를 이용하여 객체 직렬화 시 static, transient로 선언되어 있는 필드는 제외되는 반면에 Externalizable 인터페이스를 이용하여 객체 직렬화 시에는 writeExternal, readExternal 메소드를 구현하여 직렬화 대상의 필드를 직접 지정한다. 만약 name,age 순으로 직렬화 하여 write 한다면 read 시에도 name,age 순으로 역직렬화를 해와야 한다는 것을 알아야 한다. 순서를 지키지 않는 다면 java.io.EOFException 예외가 발생한다.

 

경축! 아무것도 안하여 에스천사게임즈가 새로운 모습으로 재오픈 하였습니다.
어린이용이며, 설치가 필요없는 브라우저 게임입니다.
https://s1004games.com

마지막으로 객체 직렬화 시 serialVersionUID가 Serializable 인터페이스를 구현한 클래스에 선언되어 있어야 한다. 명시적으로 선언하지 않았다면 JVM에서 자동 생성해 준다.

 

serialVersionUID가 필요한 이유는 무엇일까?

직렬화 과정에서 serialVersionUID의 버전이 포함되게 되고, 역직렬화 과정에서 java class에 선언되어 있는 serialVersionUID의 버전과 서로 동일한지 체크를 하게 된다.

 

serialVersionUID 버전 체크를 하는 이유는?

버전이 바뀌게 되면 객체의 상태가 바뀌었다는 것과 같은 의미이고, 결국 역직렬화 과정에서 오류가 발생하기 때문이다.

 

버전이 맞지 않으면 다음과 같은 예외 발생

java.io.InvalidClassException: serializable.AdminUser; local class incompatible: stream classdesc serialVersionUID = 1, local class serialVersionUID = 3

 

 

 

  package serializable;
   
  import static org.hamcrest.CoreMatchers.*;
  import static org.junit.Assert.*;
   
  import java.io.Externalizable;
  import java.io.File;
  import java.io.FileInputStream;
  import java.io.FileOutputStream;
  import java.io.IOException;
  import java.io.ObjectInput;
  import java.io.ObjectInputStream;
  import java.io.ObjectOutput;
  import java.io.ObjectOutputStream;
  import java.io.Serializable;
   
  import lombok.Data;
   
  import org.junit.Before;
  import org.junit.Test;
   
  public class Serializer {
   
  private String filePath;
  private String externalFilePath;
  private User user;
  private AdminUser adminUser;
   
  @Before
  public void init() {
  filePath = "E:\test\serializable\user.ser";
  externalFilePath = "E:\test\serializable\adminUser.ser";
   
  user = new User();
  user.setName("kyu");
  user.setAge(32);
  user.setSocialNumber("222222-1111111");
   
  adminUser = new AdminUser();
  adminUser.setName("adminKyu");
  adminUser.setAge(33);
  user.setSocialNumber("121212121-121212121");
  }
   
  @Test
  public void 객체직렬화테스트() throws IOException {
  FileOutputStream fos = new FileOutputStream(filePath);
  ObjectOutputStream oos = new ObjectOutputStream(fos);
  oos.writeObject(user);
  oos.close();
   
  assertThat(true, is(new File(filePath).isFile()));
  }
   
  @Test
  public void 객체역질렬화테스트() throws IOException, ClassNotFoundException {
  FileInputStream fis = new FileInputStream(filePath);
  ObjectInputStream ois = new ObjectInputStream(fis);
  User user = (User) ois.readObject();
  ois.close();
   
  assertThat("kyu", is(user.getName()));
  assertThat(32, is(user.getAge()));
  assertThat(null, is(user.getSocialNumber()));
  }
   
  @Test
  public void externalWrite() throws IOException {
  FileOutputStream fos = new FileOutputStream(externalFilePath);
  ObjectOutputStream oos = new ObjectOutputStream(fos);
  oos.writeObject(adminUser);
  oos.close();
   
  assertThat(true, is(new File(externalFilePath).isFile()));
  }
   
  @Test
  public void externalRead() throws IOException, ClassNotFoundException {
  FileInputStream fis = new FileInputStream(externalFilePath);
  ObjectInputStream ois = new ObjectInputStream(fis);
  AdminUser adminUser = (AdminUser) ois.readObject();
   
  assertThat("adminKyu", is(adminUser.getName()));
  assertThat(33, is(adminUser.getAge()));
  assertThat(null, is(adminUser.getSocialNumber()));
  }
  }
   
  @Data
  class User implements Serializable {
  private static final long serialVersionUID = 1L;
  private String name;
  private int age;
  private transient String socialNumber;
  }
   
  @Data
  class AdminUser implements Externalizable {
  private static final long serialVersionUID = 1L;
  private String name;
  private int age;
  private String socialNumber;
   
  @Override
  public void writeExternal(ObjectOutput out) throws IOException {
  out.writeObject(name);
  out.writeInt(age);
  }
   
  @Override
  public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException {
  this.name = (String) in.readObject();
  this.age = in.readInt();
  }
  }
view raw Serializer.java hosted with ❤ by GitHub

 

저작자 표시

 

 

 

 

본 웹사이트는 광고를 포함하고 있습니다.
광고 클릭에서 발생하는 수익금은 모두 웹사이트 서버의 유지 및 관리, 그리고 기술 콘텐츠 향상을 위해 쓰여집니다.
번호 제목 글쓴이 날짜 조회 수
164 Java 8 개선 사항 관련 글 모음 졸리운_곰 2016.05.23 271
163 Building a Search Engine With Nutch Solr And Hadoop file 졸리운_곰 2016.04.21 219
162 Nutch and Hadoop Tutorial file 졸리운_곰 2016.04.21 209
161 Latest step by Step Installation guide for dummies: Nutch 0. file 졸리운_곰 2016.04.21 131
160 Nutch 초간단 빌드와 실행 졸리운_곰 2016.04.21 457
159 Nutch로 알아보는 Crawling 구조 - Joinc 졸리운_곰 2016.04.21 352
158 A tiny bittorrent library Java: 자바로 만든 작은 bittorrent 라이브러리 file 졸리운_곰 2016.04.20 242
157 Updating UI in Eclipse RCP 졸리운_곰 2015.11.07 183
156 Eclipse RCP: Display.getDefault().asyncExec still blocking my GUI 졸리운_곰 2015.11.07 135
155 The Eclipse RCP multi-threaded Job and UIJob, use 졸리운_곰 2015.11.07 243
154 SWT: Respond to Long Running Tasks 졸리운_곰 2015.11.07 91
153 SWT and multithreading 졸리운_곰 2015.11.07 73
152 Eclipse RCP multithreading 졸리운_곰 2015.11.07 146
151 Eclipse RCP - Prevent UI freezing while running long operation 졸리운_곰 2015.11.07 114
150 Eclipse RCP Tutorial: How to Add a Progress Bar file 졸리운_곰 2015.11.07 119
149 Eclipse RCP 에서 configuration 설정 파일 사용하지 않기 secret 졸리운_곰 2015.11.07 0
148 Gson User Guide 졸리운_곰 2015.10.26 318
147 Java Google Json (Gson) Type Adapter 졸리운_곰 2015.10.26 145
146 TypeAdapter (Gson 2.3.1 API) - Google Gson 졸리운_곰 2015.10.26 132
145 Gson TypeAdapter Example file 졸리운_곰 2015.10.26 86
대표 김성준 주소 : 경기 용인 분당수지 U타워 등록번호 : 142-07-27414
통신판매업 신고 : 제2012-용인수지-0185호 출판업 신고 : 수지구청 제 123호 개인정보보호최고책임자 : 김성준 sjkim70@stechstar.com
대표전화 : 010-4589-2193 [fax] 02-6280-1294 COPYRIGHT(C) stechstar.com ALL RIGHTS RESERVED