>Java >java지도 시간 >Java 기본 - 직렬화

Java 기본 - 직렬화

PHP中文网
PHP中文网원래의
2017-07-09 18:13:101508검색

一、java中的序列化

当两个进程在进行远程通信时,彼此可以发送各种类型的数据。无论是何种类型的数据,都会以二进制序列的形式传送。发送方需要把这个Java对象转换为字节序列,才能在网络上传送;接收方则需要把字节序列再恢复为Java对象。把Java对象转换为字节序列的过程称为对象的序列化。把字节序列恢复为Java对象的过程称为对象的反序列化。

对象的序列化主要有两种用途:

1) 把对象的字节序列永久地保存到硬盘上,通常存放在一个文件中;

2) 在网络上传送对象的字节序列。

JDK类库中的序列化API

ObjectOutputStream //代表对象输出流,它的writeObject(Object obj)方法可对参数指定的obj对象进行序列化,把得到的字节序列写到一个目标输出流中。

ObjectInputStream  //代表对象输入流,它的readObject()方法从一个源输入流中读取字节序列,再把它们反序列化为一个对象,并将其返回。

只有实现了Serializable和Externalizable接口的类的对象才能被序列化。

Externalizable接口继承自Serializable接口,实现Externalizable接口的类完全由自身来控制序列化的行为,而仅实现Serializable接口的类可以采用默认的序列化方式
对象序列化包括如下步骤:

1) 创建一个对象输出流,它可以包装一个其他类型的目标输出流,如文件输出流;

2) 通过对象输出流的writeObject()方法写对象。

对象反序列化的步骤如下:

1) 创建一个对象输入流,它可以包装一个其他类型的源输入流,如文件输入流;

2) 通过对象输入流的readObject()方法读取对象。

<span style="color: #0000ff">public</span> <span style="color: #0000ff">class</span><span style="color: #000000"> Test3 {
            </span><span style="color: #0000ff">public</span> <span style="color: #0000ff">static</span> <span style="color: #0000ff">void</span> main(String[] args) <span style="color: #0000ff">throws</span><span style="color: #000000">  IOException, ClassNotFoundException {
                </span><span style="color: #008000">/*</span><span style="color: #008000">ObjectOutputStream out=new ObjectOutputStream(new FileOutputStream("c:\\1.obj"));
                
                //序列化对象
                Person p=new Person(777,"我们要好好学习");
                
                out.writeObject("我是String类型的数据");  //写入String类型
                out.writeObject(new Date()); //写入日期类型
                out.writeObject(p); //写入自定义类型
                out.writeInt(123); //定入基本数据类型
                
                out.close();</span><span style="color: #008000">*/</span>
                
                <span style="color: #008000">//</span><span style="color: #008000">反序列化对象</span>
                ObjectInputStream in=<span style="color: #0000ff">new</span> ObjectInputStream(<span style="color: #0000ff">new</span> FileInputStream("c:\\1.obj"<span style="color: #000000">));
                
                System.out.println(</span>"obj1="+<span style="color: #000000">in.readObject());  
                System.out.println(</span>"obj2="+<span style="color: #000000">in.readObject());  
                System.out.println(</span>"obj3="+<span style="color: #000000">in.readObject());  
                System.out.println(</span>"obj4="+<span style="color: #000000">in.readInt());     
                
                in.close();
            }
        }</span>
<span style="color: #008000">//</span><span style="color: #008000">Serializable 是个空接口</span>
        <span style="color: #0000ff">class</span> Person <span style="color: #0000ff">implements</span><span style="color: #000000"> Serializable{
            </span><span style="color: #0000ff">int</span><span style="color: #000000"> age;
            String name;
            
            Person(</span><span style="color: #0000ff">int</span><span style="color: #000000"> age,String name){
                </span><span style="color: #0000ff">this</span>.age=<span style="color: #000000">age;
                </span><span style="color: #0000ff">this</span>.name=<span style="color: #000000">name;
            }
            
            </span><span style="color: #0000ff">public</span><span style="color: #000000"> String toString(){
                </span><span style="color: #0000ff">return</span> <span style="color: #0000ff">this</span>.name+":"+<span style="color: #0000ff">this</span><span style="color: #000000">.age;
            }
        }</span>

2. 직렬화 가능 인터페이스 및 외부화 가능 인터페이스 (다음은 기본 지식 기록입니다)
1.Serialized

ObjectOutputStream은 직렬화 가능 인터페이스를 사용하여 클래스의 객체만 직렬화할 수 있습니다.

기본적으로 ObjectOutputStream은 기본 메서드에 따라 직렬화됩니다. 이 직렬화 메서드는 개체의 비일시적(일시적, 비지속적) 인스턴스 변수만 직렬화하고 개체의 임시 인스턴스 변수는 직렬화하지 않습니다. 또한 직렬화되지 않았습니다.

ObjectOutputStream이 기본 방식으로 역직렬화되면 다음과 같은 특징이 있습니다.

1) 메모리에 객체가 속한 클래스가 로드되지 않은 경우 이 클래스가 먼저 로드되어 초기화됩니다. 해당 클래스 파일이 클래스 경로에 없으면 ClassNotFoundException이 발생합니다.

2) 역직렬화 중에는 클래스의 생성자 메서드가 호출되지 않습니다.

사용자가 클래스 직렬화 방법을 제어하려는 경우 직렬화 가능 클래스에서 다음 형식의 writeObject() 및 readObject() 메서드를 제공할 수 있습니다.

private void writeObject(java.io.ObjectOutputStream out)가 IOException을 발생시킵니다

private void readObject(java.io.ObjectInputStream in)는 IOException, ClassNotFoundException을 발생시킵니다.

ObjectOutputStream이 Person 객체를 직렬화할 때 객체에 writeObject() 메서드가 있으면 이 메서드가 실행되고, 그렇지 않으면 기본 방식으로 직렬화됩니다. 객체의 writeObjectt() 메서드에서 먼저 ObjectOutputStream의 defaultWriteObject() 메서드를 호출하여 객체 출력 스트림이 먼저 기본 직렬화 작업을 수행하도록 할 수 있습니다. 역직렬화의 경우에도 마찬가지이지만 이번에는 defaultReadObject() 메서드가 호출됩니다.

일부 개체에는 대중에게 공개해서는 안 되는 민감한 정보가 포함되어 있습니다. 기본 방식으로 직렬화되면 네트워크를 통해 전송될 때 직렬화된 데이터가 범죄자에 의해 도난당할 수 있습니다. 이러한 유형의 정보는 암호화된 후 직렬화될 수 있으며, 역직렬화 중에 암호를 해독하고 원래 정보로 복원해야 합니다.

기본 직렬화 방법은 전체 개체 그래프를 직렬화하므로 개체 그래프를 재귀적으로 순회해야 합니다. 객체 그래프가 복잡한 경우 재귀 순회 작업은 많은 공간과 시간을 소비하며 내부 데이터 구조는 양방향 목록입니다.

적용 중에 일부 멤버 변수를 임시 유형(private temporary int age)으로 변경하면 공간과 시간이 절약되고 직렬화 성능이 향상됩니다.

2.외부화 가능

직렬화 가능 인터페이스에서 상속됨 ​​클래스가 외부화 가능 인터페이스를 구현하는 경우 이 클래스는 자체 직렬화 동작을 완전히 제어합니다. 외부화 가능 인터페이스는 두 가지 메소드를 선언합니다:

public void writeExternal(ObjectOutput out)이 IOException을 발생시킵니다

public void readExternal(ObjectInput in)은 IOException, ClassNotFoundException

을 발생시킵니다.

전자는 직렬화 작업을 담당하고 후자는 역직렬화 작업을 담당합니다.

외부화 가능 인터페이스를 구현하는 클래스의 객체를 역직렬화할 때 /*는 먼저 매개 변수 없이 클래스의 생성자를 호출합니다. 이는 기본 역직렬화 방법과 다릅니다 */

클래스의 매개변수가 없는 생성자를 삭제하거나 생성자의 접근권한을 private, default, protected 수준으로 설정하면

java.io.InvalidException이 발생합니다. 유효한 생성자 예외가 없습니다.

3.serialVersionUID

It직렬화 가능 클래스의 다양한 버전의 직렬화 호환성. 직렬화 가능 인터페이스를 구현하는 모든 클래스에는 직렬화 버전 식별자를 나타내는 정적 변수가 있습니다.

개인 정적 최종 긴 serialVersionUID;

위의 serialVersionUID 값은 클래스의 내부 세부 정보를 기반으로 Java 런타임 환경에서 자동으로 생성됩니다. 클래스의 소스 코드를 수정한 후 다시 컴파일하면 새로 생성된 클래스 파일의 serialVersionUID 값도 변경될 수 있습니다.

클래스의 serialVersionUID 기본값은 Java 컴파일러의 구현에 따라 완전히 다릅니다. 다른 Java 컴파일러로 컴파일된 동일한 클래스의 경우 다른 serialVersionUID가 생성되거나 동일할 수 있습니다. serialVersionUID의 독립성과 확실성을 향상하려면 직렬화 가능 클래스에서 serialVersionUID를 명시적으로 정의하고 명시적인 값을 제공하는 것이 좋습니다. serialVersionUID를 명시적으로 정의하는 데에는 두 가지 용도가 있습니다.

1) 어떤 경우에는 클래스의 서로 다른 버전이 직렬화와 호환되기를 바라므로 클래스의 서로 다른 버전이 동일한 serialVersionUID를 갖도록 해야 합니다.

2) 경우에 따라 클래스의 서로 다른 버전이 직렬화 호환되는 것을 원하지 않으므로 클래스의 서로 다른 버전이 서로 다른 serialVersionUID를 갖도록 해야 합니다.

위 내용은 Java 기본 - 직렬화의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
이전 기사:Java 기본 iO 소개다음 기사:Java 기본 iO 소개