ObjectOutputStream 追加写入读取错误

问题描述:

用类ObjectOutputStream向文件写读对象时,碰到一个问题:新建一个文件,用输出流ObjectOutputStream向文件连续写几个对象,关闭输出流,然 后读取,这些对象都可以读出;这时在向该文件增加对象,新写的对象就读不出了

问题出现的原因:

ObjectOutputStream建立后第一次写入一个对象时, 会在对象数据前写入一些标志的数据“AC ED  00 05”(用二进制方式查看打开),应该是流相关的信息。当你关闭 ObjectOutputStream 后再重新打开往文件里面写对象时(append方式),就会再一次把“AC ED 00 05”写入文件,而这些信息并不是你写入对象的数据,所以当你用ObjectInputStream来读取对象时,流会将除第一个“AC ED 00 05”以外的数据当作各个对象的数据,造成无法解析,所以读取时有一个java.io.StreamCorruptedException出现。 这个可以通过编辑Info.dat来验证,只要将“AC ED 00 05”删除(第一个“AC ED 00 05”保留)就可以正常读出后来加入的对象。 给出一个比较笨的解决方法: 在以后要添加新的对象到Info.dat时,将里面原有的对象读出放入ArrayList中,清空文件,再将对象集一次写入。 

尝试解决办法: 

那个“AC ED 00 05”是  ObjectOutputStream.writeSystemHeader()写进去的,你可以继承ObjectOutputStream类,覆盖这个方法。 在你自己的writeSystemHeader()里判断是不是第一次写入一个文件,如果是向一个文件大小不为零的文件追加的话,就调用  super.reset(),如果是第一次写这个文件,不是追加,就调用super.writeSystemHeader() 

自定义类MyObjectOutputStream 

	/**
	 * 此类继承ObjectOutputStream,重写writeStreamHeader()方法,以实现追加写入时去掉头部信息
	 */
	public static class MyObjectOutputStream extends ObjectOutputStream {
		private static File f;

		// writeStreamHeader()方法是在ObjectOutputStream的构造方法里调用的
		// 由于覆盖后的writeStreamHeader()方法用到了f。如果直接用此构造方法创建
		// 一个MyObjectOutputStream对象,那么writeStreamHeader()中的f是空指针
		// 因为f还没有初始化。所以这里采用单态模式
		private MyObjectOutputStream(OutputStream out, File f) throws IOException, SecurityException {
			super(out);
		}

		// 返回一个MyObjectOutputStream对象,这里保证了new MyObjectOutputStream(out, f)
		// 之前f已经指向一个File对象
		public static MyObjectOutputStream newInstance(File file, OutputStream out) throws IOException {
			f = file;// 本方法最重要的地方:构建文件对象,两个引用指向同一个文件对象
			return new MyObjectOutputStream(out, f);
		}

		@Override
		protected void writeStreamHeader() throws IOException {
			// 文件不存在或文件为空,此时是第一次写入文件,所以要把头部信息写入。
			if (!f.exists() || (f.exists() && f.length() == 0)) {
				super.writeStreamHeader();
			} else {
				// 不需要做任何事情
			}
		}
	}


测试方法: 

	public static void writeObject() throws Exception {
		String filename = "test.txt";
		File f = new File(filename);
		OutputStream os = new FileOutputStream(f, true);
		MyObjectOutputStream moos = MyObjectOutputStream.newInstance(f, os);
		moos.writeObject("abc");
		moos.writeObject(1);
		moos.close();
	}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值