使用ObjectOutputStream,如果以追加的方式,在一个文件中末尾追加写入object,那每次都会在写入时增加特定的文件头信息,而这些头信息如果每次都写入到文件中,在使用ObjectInputStream
读取这个文件object时就会出现异常.
下面的是自己实现的追加写入Object信息到文件中类:
java.io.StreamCorruptedException: invalid type code: AC
而文件的头信息格式是这样的:
AC ED 00 05
这个头信息只能出现在整个文件的头部,且仅出现一次.
所以在一个文件中追加写入Object,就需要自定义自己的类来实现.
package com.xjl456852.process;
import org.junit.Test;
import javax.script.ScriptException;
import java.io.*;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Created by xjl456852 on 2016/3/11.
*/
public class ObjectTest {
/**
* 使用ObjectOutputStream写Object对象,不追加的情况,这个是可以的.
* @throws ScriptException
* @throws IOException
*/
@Test
public void testObject() throws ScriptException, IOException {
ObjectOutputStream oos = null;
ObjectInputStream ois = null;
try {
String file = "d:/xjl/object.txt";
//使用ObjectOutputStream追加写入,在多次写完之后,读取object信息时会有异常.所以不能这样写.
// oos = new ObjectOutputStream(new FileOutputStream(new File(file),true));
oos = new ObjectOutputStream(new FileOutputStream(new File(file)));
List<Map<Integer,String>> list = new ArrayList<Map<Integer,String>>();
Map<Integer, String> map = new HashMap<>();
map.put(1, "xjl");
map.put(2, "xjl");
list.add(map);
oos.writeObject(list);
oos.flush();
List<Map<Integer,String>> list2 = new ArrayList<Map<Integer,String>>();
Map<Integer, String> map2 = new HashMap<>();
map2.put(3, "xjl3");
map2.put(4, "xjl4");
list2.add(map2);
oos.writeObject(list2);
oos.flush();
ois = new ObjectInputStream(new FileInputStream(new File(file)));
List<Map<Integer,String>> listResult = (List<Map<Integer,String>>)ois.readObject();
List<Map<Integer,String>> listResult2 = (List<Map<Integer,String>>)ois.readObject();
printList(listResult);
printList(listResult2);
} catch(Exception e) {
e.printStackTrace();
} finally {
if(oos != null) {
oos.close();
oos = null;
}
if(ois != null) {
ois.close();
ois = null;
}
}
}
/**
* 使用自定义的方式,在同一个文件中追加Object,要使用下面的方式
* @throws ScriptException
* @throws IOException
*/
@Test
public void testObject2() throws ScriptException, IOException {
ObjectInputStream ois = null;
try {
String file = "d:/xjl/object2.txt";
List<Map<Integer,String>> list = new ArrayList<Map<Integer,String>>();
Map<Integer, String> map = new HashMap<>();
map.put(1, "xjl");
map.put(2, "xjl");
list.add(map);
writeObject(list,file);
List<Map<Integer,String>> list2 = new ArrayList<Map<Integer,String>>();
Map<Integer, String> map2 = new HashMap<>();
map2.put(3, "xjl3");
map2.put(4, "xjl4");
map2.put(5, "xjl4");
list2.add(map2);
writeObject(list2,file);
ois = new ObjectInputStream(new FileInputStream(new File(file)));
List<Map<Integer,String>> listResult = (List<Map<Integer,String>>)ois.readObject();
List<Map<Integer,String>> listResult2 = (List<Map<Integer,String>>)ois.readObject();
printList(listResult);
printList(listResult2);
} catch(Exception e) {
e.printStackTrace();
} finally {
if(ois != null) {
ois.close();
ois = null;
}
}
}
/**
*
*
* 如果在文件中追加对象,每次创建ObjectOutputStream,都会在每次写入时增加header信息.
* 导致读取时出现java.io.StreamCorruptedException: invalid type code: AC异常
* @param t
* @param file
* @param <T>
* @throws IOException
*/
public <T>void writeObject (T t, String file) throws IOException {
ObjectOutputStream oos = null;
try {
oos = MyOutputStream.newInstance(file);
oos.writeObject(t);
oos.flush();
} catch (Exception e) {
e.printStackTrace();
} finally {
if(oos != null) {
oos.close();
oos = null;
}
}
}
public static void printList(List<Map<Integer, String>> list) {
for(Map<Integer,String> map : list) {
for(Map.Entry<Integer,String > entry : map.entrySet()) {
System.out.println(entry.getKey() +":"+ entry.getValue());
}
}
}
}
下面的是自己实现的追加写入Object信息到文件中类:
package com.xjl456852.process;
import java.io.*;
/**
* Created by xjl456852 on 2016/3/11.
*/
/**
* 使用自定义的方式,来实现追加写入object
*/
public class MyOutputStream extends ObjectOutputStream {
public MyOutputStream(OutputStream out) throws IOException {
super(out);// 会调用writeStreamHeader()
}
/**
* 覆盖父类的方法,使其在已有对象信息并追加时,不写header信息
* 查看源码会发现:writeStreamHeader方法会写入以下两行内容:
*
* bout.writeShort(STREAM_MAGIC);
* bout.writeShort(STREAM_VERSION);
*
* 这两行对应的值:
* final static short STREAM_MAGIC = (short)0xaced;
* final static short STREAM_VERSION = 5;
*
* 在文件中头部就会写入:AC ED 00 05
* 一个文件对象只有在文件头出应该出现此信息,文件内容中不能出现此信息,否则会导致读取错误
* 所以在追加时,就需要覆盖父类的writeStreamHeader方法,执行reset()方法
*
* reset()方法写入的是这个:final static byte TC_RESET = (byte)0x79;
* @throws IOException
*/
@Override
protected void writeStreamHeader() throws IOException {
super.reset();
}
public static ObjectOutputStream newInstance(String file) throws IOException {
File f = new File(file);
long length = f.length();
ObjectOutputStream oos = null;
if(f.length() == 0) {
oos = new ObjectOutputStream(new FileOutputStream(new File(file),true));
} else {
oos = new MyOutputStream(new FileOutputStream(new File(file),true));
}
return oos;
}
}