关于java的序列化

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了关于java的序列化相关的知识,希望对你有一定的参考价值。

如果使用的类没有实现Serializable接口,而且这个类又来自于不可更改的工具包,如何将其序列化以用来进行套接字网络传输?
可能我说的不太清楚。通过我的代码来说明吧:
我是用的是opensaml这个中间件。其中有XMLObject这个借口。现在我想让:
XMLObject obj;
这个变量序列化之后进行网络传输。但是XMLOBject在定义的时候并没有实现Serializable接口。
总之我想序列化的是obj这个对象,不是其中的成员变量。
麻烦大家了~

首先说明一下序列化的知识:java中的序列化(serialization)机制能够将一个实例对象的状态信息写入到一个字节流中,使其可以通过socket进行传输、或者持久化存储到数据库或文件系统中;然后在需要的时候,可以根据字节流中的信息来重构一个相同的对象序列化机制在java中有着广泛的应用3EJB、RMI等技术都是以此为基础的。序列化机制是通过java.io.ObjectOutputStream类和java.io.ObjectInputStream类来实现的。在序列化(serialize)一个对象的时候l会先实例化一个ObjectOutputStream对象,然后调用其writeObject()方法;在反序列化(deserialize)的时候,则会实例化一个ObjectInputStream对象,然后调用其readObject()方法。上面您的错误,就是在于有一个或者几个没有"序列化"的数据,导致没有办法创建输出流,导致发生的java.io.NotSerializableException。之所以要序列化,我猜测是因为您的数据里面存在一个对象型的数据,但是该对象没有实现序列化。比如:您有一个字段为address,这个字段您是通过一个类Address来描述的,Address里面可能有province、city、street等等属性或者一些setter 和getter,如果这个类,没有实现序列化,往往会出现这个问题毕竟没有看到程序,是我的一个猜测,请检查一下程序或者发出来进行进一步讨论。追问

转至声明:

让后我就不知道怎么办了...

参考技术A 那要看这个类的成员是啥样的,如果存在private成员那就没法处理了,protected和public成员可以通过继承读出,但多个类时就要继承出多个,也可通过反射机制读出,反射机制可对多个类统一处理,追问

可能我说的不太清楚。通过我的代码来说明吧:
我是用的是opensaml这个中间件。其中有XMLObject这个借口。现在我想让:
XMLObject obj;
这个变量序列化之后进行网络传输。但是XMLOBject在定义的时候并没有实现Serializable接口。
总之我想序列化的是obj这个对象,不是其中的成员变量。
麻烦了~

追答

不好意思,我没用过opensaml,这个提不了建议了

参考技术B 写个类A继承该类,序列化类A即可追问

但是怎样才可以把父类转换成自己写的子类A

可能我说的不太清楚。通过我的代码来说明吧:
我是用的是opensaml这个中间件。其中有XMLObject这个借口。现在我想让:
XMLObject obj;
这个变量序列化之后进行网络传输。但是XMLOBject在定义的时候并没有实现Serializable接口。
总之我想序列化的是obj这个对象,不是其中的成员变量。
麻烦了~

关于java中的对象序列化

java对象序列化机制一般来讲有两种用途:

1.需要将对象的状态保存到文件中,而后能够通过读入对象状态来重新构造对象,恢复程序状态

2.使用套接字在网络上传送对象的程序来说,是很有用的。

我们通过让类实现java.io.Serializable 接口可以将类序列化。这个接口是一个制造者(marker)接口。也就是说,对于要实现它的类来说,该接口不需要实现任何方法。它主要用来通知Java虚拟机(JVM),需要将一个对象序列化。

对于这个,有几点我们需要明确:

1.并非所有类都可以序列化,在cmd下,我们输入serialver java.net.socket,可以得到socket是否可序列化的信息,实际上socket是不可序列化的。

2.java有很多基础类已经实现了serializable接口,比如string,vector等。但是比如hashtable就没有实现serializable接口。

将对象读出或者写入流的主要类有两个: ObjectOutputStream与ObjectInputStream 。ObjectOutputStream 提供用来将对象写入输出流的writeObject方法, ObjectInputStream提供从输入流中读出对象的readObject方法。使用这些方法的对象必须已经被序列化的。也就是说,必须已经实现Serializable接口。如果你想writeobject一个hashtable对象,那么,会得到一个异常。

下面举个例子:

import java.io.*;

public class testser implements Serializable {

public int ii;

testser() {

}

testser( int param ) {

ii = param;

}

}

testser是一个实现了serializable接口的类。

读写这个序列化过的类:

import java.io.*;

public class Ser {

private static String datafile;

datafile="ser.data";

public static void main( String[] argv ) {

System.out.println( "Java Serialization Demo." );

SerData data;

try {

ObjectInputStream in = new ObjectInputStream( new FileInputStream( datafile ));

data = (SerData) in.readObject();

in.close();

}

catch (Exception e) {

data = new testser();

}

System.out.println( "Original data: ii = " + data.ii );

data.ii++;

try {

ObjectOutputStream out = new ObjectOutputStream( new FileOutputStream( datafile ) );

out.writeObject( data );

out.flush();

out.close();

}

catch (Exception e) {

System.out.println( e );

}

}

}

还有可以通过套接字传递序列化对象,大概类似,不再赘述。

以上是关于关于java的序列化的主要内容,如果未能解决你的问题,请参考以下文章

关于java的序列化

夯实Java基础系列27:关于Java序列化的13个面试问题

关于java Serializable接口的问题

关于java的Serializable

关于Java序列化的问题你真的会吗?

关于Java序列化的问题你真的会吗?