我的Java开发学习之旅------>Java使用ObjectOutputStream和ObjectInputStream序列号对象报java.io.EOFException异常的解决方法

今天用ObjectOutputStream和ObjectInputStream进行对象序列化话操作的时候,报了java.io.EOFException异常。

异常代码如下:

java.io.EOFException
	at java.io.ObjectInputStream$BlockDataInputStream.peekByte(ObjectInputStream.java:2554)
	at java.io.ObjectInputStream.readObject0(ObjectInputStream.java:1297)
	at java.io.ObjectInputStream.readObject(ObjectInputStream.java:351)
	at cn.fuxi.io.ReadObject.readObject(ReadObject.java:27)
	at cn.fuxi.io.ReadObject.main(ReadObject.java:12)

这个异常是因为我使用ObjectInputStream读取对象时候,没有判断好ObjectInputStream是否读取到了未知长度的文件末尾,导致到了文件末尾,程序无法正常读取文件内容。

后来研究了一下以后,得出三种解救思路:

第一种方法:

在写完对象后,加一句 oos.writeObject(null);  插入null是用来判断是否读取到结尾。oos是ObjectOutputStream实例。

  然后读对象的时候,使用while ((obj = ois.readObject()) != null),来判断再次读取一个对象,判定对象是否是空,如果不为null继续读取,如果为null停止读取。

第二种方法:

将若干个对象(数量不定)都装入 一个容器中(如:ArrayList之类的),然后写对象的时候,将该容器写入。

读取的时候,读取出容器,然后再遍历容器,取出自己所需的对象。

第三个方法:

和第二个方法类似,将对象都存入数组中,然后写入数组对象。

读取的时候,取出数组,再遍历数组,取出所需的对象。

下面是具体的代码操作:

1、要进行序列号的对象  Person.java

import java.io.Serializable;

public class Person implements Serializable {
	/**
	 *
	 */
	private static final long serialVersionUID = -6374324573857634276L;
	private String name;
	private int age;

	public Person(String name, int age) {
		this.name = name;
		this.age = age;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getAge() {
		return age;
	}

	public void setAge(int age) {
		this.age = age;
	}

}

2、写入对象的三种方法    

import java.io.FileOutputStream;
import java.io.ObjectOutputStream;
import java.util.ArrayList;
import java.util.List;

public class WriteObject {

	/**
	 * @param args
	 */
	public static void main(String[] args) {
		writeObjectByArray();
		writeObject();
		writeObjectByList();
	}
	/**
	 * 直接写入对象
	 */
	private static void writeObject() {
		try {
			ObjectOutputStream oos = new ObjectOutputStream(
					new FileOutputStream("object.txt"));
			for (int i = 1; i < 10; i++) {
				oos.writeObject(new Person("欧阳鹏[" + i+"]", i));
			}
			//插入null是用来判断是否读取到结尾
			//写入结束标志方便读取(非常重要,如果不写入,在读取的时候无法定位读取结束);
			//读取的时候就会报  java.io.EOFException  异常
			oos.writeObject(null);
			oos.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	/**
	 * 通过数组写入对象
	 */
	private static void writeObjectByArray() {
		try {
			ObjectOutputStream oos = new ObjectOutputStream(
					new FileOutputStream("objectByArray.txt"));
			Person[] persons = new Person[10];
			for (int i = 1; i < 10; i++) {
				Person person = new Person("欧阳鹏 Arrays[" + (10+i)+"]", 10+i);
				persons[i] = person;
			}
			oos.writeObject(persons);
			oos.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	/**
	 * 通过集合写入对象
	 */
	private static void writeObjectByList() {
		try {
			ObjectOutputStream oos = new ObjectOutputStream(
					new FileOutputStream("objectByList.txt"));
			List<Person> persons=new ArrayList<Person>();
			for (int i = 1; i < 10; i++) {
				Person person = new Person("欧阳鹏 List[" + (20+i)+"]", 20+i);
				persons.add(person);
			}
			//写入List
			oos.writeObject(persons);
			oos.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}

3、读取对象的三种方法    

import java.io.FileInputStream;
import java.io.ObjectInputStream;
import java.util.List;

public class ReadObject {
	/**
	 * @param args
	 */
	public static void main(String[] args) {
		readObject();
		System.out.println("==============================");
		readObjectByArrays();
		System.out.println("==============================");
		readObjectByList();
	}
	/**
	 * 直接读取对象
	 */
	private static void readObject() {
		try {
			ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
					"object.txt"));
			Object obj = null;
			//如果为null就读取到文件结尾了
			//读取结束标志位:是再次读取一个对象,判定对象是否是空,如果不为null继续读取,如果为null停止读取
			while ((obj = ois.readObject()) != null) {
				////反序列化读取得到Person
				Person person = (Person) obj;
				System.out.println("名字为:" + person.getName() + "  年龄为:"
						+ person.getAge());
			}
			ois.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	/**
	 * 通过数组读取对象
	 */
	private static void readObjectByArrays() {
		try {
			ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
					"objectByArray.txt"));
			//反序列化读取得到Person[]
			Person[] persons = (Person[]) ois.readObject();
			for (int i = 1; i < persons.length; i++) {
				Person person = (Person) persons[i];
				System.out.println("名字为:" + person.getName() + "  年龄为:"
						+ person.getAge());
			}
			ois.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
	/**
	 * 通过集合读取对象
	 */
	private static void readObjectByList() {
		try {
			ObjectInputStream ois = new ObjectInputStream(new FileInputStream(
					"objectByList.txt"));
			//反序列化读取得到List<Person>
			List<Person> persons = (List<Person>) ois.readObject();
			for (int i = 0; i <persons.size(); i++) {
				Person person=persons.get(i);
				System.out.println("名字为:" + person.getName() + "  年龄为:"
						+ person.getAge());
			}
			ois.close();
		} catch (Exception e) {
			e.printStackTrace();
		}
	}

}

下面是结果:

写入对象后,在src同级目录下,生成3个文件,分别是object.txt 、objectByArray.txt、objectByList.txt   如下图所示。

读取对象,结果如下:

名字为:欧阳鹏[1]  年龄为:1
名字为:欧阳鹏[2]  年龄为:2
名字为:欧阳鹏[3]  年龄为:3
名字为:欧阳鹏[4]  年龄为:4
名字为:欧阳鹏[5]  年龄为:5
名字为:欧阳鹏[6]  年龄为:6
名字为:欧阳鹏[7]  年龄为:7
名字为:欧阳鹏[8]  年龄为:8
名字为:欧阳鹏[9]  年龄为:9
==============================
名字为:欧阳鹏 Arrays[11]  年龄为:11
名字为:欧阳鹏 Arrays[12]  年龄为:12
名字为:欧阳鹏 Arrays[13]  年龄为:13
名字为:欧阳鹏 Arrays[14]  年龄为:14
名字为:欧阳鹏 Arrays[15]  年龄为:15
名字为:欧阳鹏 Arrays[16]  年龄为:16
名字为:欧阳鹏 Arrays[17]  年龄为:17
名字为:欧阳鹏 Arrays[18]  年龄为:18
名字为:欧阳鹏 Arrays[19]  年龄为:19
==============================
名字为:欧阳鹏 List[21]  年龄为:21
名字为:欧阳鹏 List[22]  年龄为:22
名字为:欧阳鹏 List[23]  年龄为:23
名字为:欧阳鹏 List[24]  年龄为:24
名字为:欧阳鹏 List[25]  年龄为:25
名字为:欧阳鹏 List[26]  年龄为:26
名字为:欧阳鹏 List[27]  年龄为:27
名字为:欧阳鹏 List[28]  年龄为:28
名字为:欧阳鹏 List[29]  年龄为:29

==================================================================================================

  作者:欧阳鹏  欢迎转载,与人分享是进步的源泉!

  转载请保留原文地址:http://blog.csdn.net/ouyang_peng

==================================================================================================

判断ObjectInputStream是否读到文件末尾,文件长度未知

判断ObjectInputStream是否读到文件末尾,文件长度未知

判断ObjectInputStream是否读到文件末尾,文件长度未知

时间: 2025-01-21 09:13:57

我的Java开发学习之旅------&gt;Java使用ObjectOutputStream和ObjectInputStream序列号对象报java.io.EOFException异常的解决方法的相关文章

我的Java开发学习之旅------&amp;gt;Java经典面试题

摘自张孝祥itcast 从享受生活的角度上来说:"程序员并不是一种最好的职业,我认为两种人可以做程序员,第一,你不做程序员,你就没有什么工作可做,或者说是即使有可以做的工作但是你非常不愿意去做:第二,你非常痴迷和爱好程序,并且在这方面有一些天赋和优势.程序员的结局也是有两种:第一,默默退休,第二以程序员为起点或跳板,注意积累,跟对了好的老板或团队,找到和很好的搭档自己创业,成为IT金领和富翁." 人们在时间面前是平等的,吾生也有涯,所以,你的经验更丰富点,那不算什么,经验是用时间积累的

我的Java开发学习之旅------&amp;gt;Java双重检查锁定及单例模式详解(转)

简介:          所有的编程语言都有一些共用的习语.了解和使用一些习语很有用,程序员们花费宝贵的时间来创建.学习和实现这些习语.问题是,稍后经过证明,一些习语并不完全如其所声称的那样,或者仅仅是与描述的功能不符.在 Java 编程语言中,双重检查锁定就是这样的一个绝不应该使用的习语.在本文中,Peter Haggar 介绍了双重检查锁定习语的渊源,开发它的原因和它失效的原因.         单例创建模式是一个通用的编程习语.和多线程一起使用时,必需使用某种类型的同步.在努力创建更有效的

我的Java开发学习之旅------&amp;gt;Java字符编码解析

Java开发中,常常会遇到乱码的问题,一旦遇到这种问题,常常就很扯蛋,每个人都不愿意承认是自己的代码有问题.其实编码问题并没有那么神秘,那么不可捉摸,搞清Java的编码本质过程就真相大白了.               其实,编码问题存在两个方面:JVM之内和JVM之外.   1.Java文件编译后形成class 这里Java文件的编码可能有多种多样,但Java编译器会自动将这些编码按照Java文件的编码格式正确读取后产生class文件,这里的class文件编码是Unicode编码(具体说是UT

我的Java开发学习之旅------&amp;gt;心得总结:Java性能优化技巧集锦

一.通用篇 "通用篇"讨论的问题适合于大多数Java应用. 1.1 不用new关键词创建类的实例 用new关键词创建类的实例时,构造函数链中的所有构造函数都会被自动调用.但如果一个对象实现了Cloneable接口,我们可以调用它的clone()方法.clone()方法不会调用任何类构造函数. 在使用设计模式(Design Pattern)的场合,如果用Factory模式创建对象,则改用clone()方法创建新的对象实例非常简单.例如,下面是Factory模式的一个典型实现: publi

我的Java开发学习之旅------&amp;gt;Java资源的国际化详解

internationalization (国际化)简称 i18n,因为在i和n之间还有18个字符,localization(本地化 ),简称L10n. 国际化相关的Java类 Java国际化主要通过如下3个类完成 java.util.ResourceBundle:用于加载一个资源包 java.util.Locale:对应一个特定的国家/区域.语言环境. java.text.MessageFormat:用于将消息格式化 国际化资源文件 为实现程序的国际化,必须提供程序所需要的资源文件.资源文件的

我的Java开发学习之旅------&amp;gt;Java NIO 报java.nio.charset.MalformedInputException: Input length = 1异常

今天在使用Java NIO的Channel和Buffer进行文件操作时候,报了java.nio.charset.MalformedInputException: Input length = 1异常,具体如下: java.nio.charset.MalformedInputException: Input length = 1 at java.nio.charset.CoderResult.throwException(CoderResult.java:260) at java.nio.char

我的Java开发学习之旅------&amp;gt;Java语言中方法的参数传递机制

       实参:如果声明方法时包含来了形参声明,则调用方法时必须给这些形参指定参数值,调用方法时传给形参的参数值也被称为实参.      Java的实参值是如何传入方法?这是由Java方法的参数传递机制来控制的,Java里方法的参数传递方式只有一种:值传递.所谓值传递,就是将实际参数的副本(复制品)传入方法内,而参数本身不会收到任何影响. 一.参数类型是原始类型的值传递 下面通过一个程序来演练 参数类型是原始类型的值传递的效果: public class ParamTransferTest

我的Java开发学习之旅------&amp;gt;工具类:Java获取字符串和文件进行MD5值

ps:这几天本人用百度云盘秒传了几部大片到云盘上,几个G的文件瞬秒竟然显示"上传成功"!这真让我目瞪口呆,要是这样的话,那得多快的网速,这绝对是不可能的,也许这仅是个假象.百度了一下才发现所谓的"秒传"是常见的"忽略式"上传方式,就是您上传了一个文件名为111.exe,MD5为一个数,有一个网友以前也上传一个叫222.exe,MD5和您上传的文件MD5码一模一样,所以这个文件上传到服务器上的时间就很短了,这是因为别人上传过这个文件,您上传这个文件

我的Java开发学习之旅------&amp;gt;Java ClassLoader解析一(转)

jvm classLoader architecture: Bootstrap ClassLoader/启动类加载器  主要负责jdk_home/lib目录下的核心 api 或 -Xbootclasspath 选项指定的jar包装入工作. Extension ClassLoader/扩展类加载器  主要负责jdk_home/lib/ext目录下的jar包或 -Djava.ext.dirs 指定目录下的jar包装入工作. System ClassLoader/系统类加载器  主要负责java -c