【问题标题】:Write and read multiple objects to file将多个对象写入和读取到文件
【发布时间】:2013-05-10 01:36:47
【问题描述】:

我正在为安卓设计一个手写应用程序。

我想在用户每次按下回车按钮时将信息 (class LogInfo) 写入日志文件。

之后,我想读取存储的信息。

这是我的课程的一部分,带有自定义写入方法:

public class LogInfo implements Serializable {

private static final long serialVersionUID = -5777674941129067422L;

public static List<Point[][]> strokes;
public static List<byte[]> codes;

// Only write and read methods shown

private void writeObject(ObjectOutputStream stream) throws IOException
{
    stream.defaultWriteObject();
    stream.writeInt(strokes.size());
    Point[][] pointsArray = null;
    for (int i = 0; i < strokes.size(); i++)
    {
        pointsArray = ((Point[][])strokes.get(i));
        stream.writeInt(pointsArray.length);
        for (int j = 0; j < pointsArray.length; j++)
        {
            stream.writeInt(pointsArray[j].length);
            for (int k = 0; k < pointsArray[j].length; k++)
            {
                stream.writeInt(pointsArray[j][k].x);
                stream.writeInt(pointsArray[j][k].y);
                //stream.writeObject(elementData[i]);
            }
        }
    }

    int size = codes.size();
    stream.writeInt(size);
    for (int i = 0; i < size; i++)
    {
        stream.write(codes.get(i));
    }
}

这是读取方法:

private void readObject(java.io.ObjectInputStream stream)
    {
        stream.defaultReadObject();
        int strokesSize = stream.readInt();
        for (int i = 0; i < strokesSize; i++)
        {
            int arrayXSize = stream.readInt();
            Point[][] points = new Point[arrayXSize][];
            for (int j = 0; j < arrayXSize; j++)
            {
                int arrayYSize = stream.readInt();
                points[j] = new Point[arrayYSize];
                for (int k = 0; k < arrayYSize; k++)
                    points[j][k] = new Point(stream.readInt(), stream.readInt());
            }
            strokes.add(points);
        }

        int codesSize = stream.readInt();
        for (int i = 0; i < codesSize; i++)
        {
            byte[] buffer = new byte[3];
            stream.read(buffer, 0, 3);
            codes.add(buffer);
        }
    }

当我在文件中只保存一个对象时效果很好。当我尝试保存更多时,阅读不起作用(它会抛出StreamCorruptedException)。 它在while循环中只读取一个对象!

在主类中,我只使用了两个简单的方法:

// WRITE TO FILE
logInfo.writeLog();

// READ FROM FILE
ArrayList<LogInfo> logInfoArrayList = logInfo.readLog();

定义为:

public void writeLog()
{
    File file = new File (Environment.getExternalStorageDirectory().getAbsolutePath(), "data.log");
    FileOutputStream fos;
    try {
        fos = new FileOutputStream(file, true);
        //fos = openFileOutput(Environment.getExternalStorageDirectory().getAbsolutePath() + "/data.log", Context.MODE_APPEND);
        ObjectOutputStream os = new ObjectOutputStream(fos);
        os.writeObject(this);
        os.close(); 
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
}

public ArrayList<LogInfo> readLog()
{
    ArrayList<LogInfo> logInfoArray = new ArrayList<LogInfo>();

    try{
        File file = new File (Environment.getExternalStorageDirectory().getAbsolutePath(), "data.log");
        FileInputStream fis  = new FileInputStream(file);
        ObjectInputStream reader = new ObjectInputStream(fis);  

        LogInfo tempLogInfo = new LogInfo();
        while((tempLogInfo = (LogInfo)reader.readObject()) != null)
            logInfoArray.add(tempLogInfo);
        reader.close();
    } catch (Exception e) {
     //TODO Auto-generated catch block
     e.printStackTrace();
    }

    return logInfoArray;
}

请求更新:

//We use this class to not write a header in a file that already exist
class MyObjectOutputStream extends ObjectOutputStream {

    public MyObjectOutputStream(OutputStream os) throws IOException {
        super(os);
      }

    @Override
    protected void writeStreamHeader() {}
}

【问题讨论】:

  • 能否发布异常堆栈跟踪
  • 05-10 10:56:40.320: W/System.err(26121): java.io.StreamCorruptedException: 格式错误:ac 05-10 10:56:40.335: W/System.err (26121): 在 java.io.ObjectInputStream.corruptStream(ObjectInputStream.java:701) 05-10 10:56:40.340: W/System.err(26121): 在 java.io.ObjectInputStream.readNonPrimitiveContent(ObjectInputStream.java: 814) 05-10 10:56:40.345: W/System.err(26121): 在 java.io.ObjectInputStream.readObject(ObjectInputStream.java:2003) 05-10 10:56:40.350: W/System.err( 26121): 在 com.myapp.LogInfo.readLog(LogInfo.java:116)
  • 循环到空是不正确的。 readObject() 在流结束时不返回 null。它抛出 EOFException。

标签: java android file object


【解决方案1】:
  1. 您不能附加到使用ObjectOutputStream 创建的现有文件,至少不能不费力。关于扩展ObjectOutputStream 和覆盖writeStreamHeader() 方法有一个技巧,以免第二次写入流标头,但我不赞成它。你真的应该重写整个文件,也许是一个列表。

  2. 您不需要所有这些代码。只需使 strokescodes 非静态和非瞬态,并完全摆脱 readObject()writeObject() 方法。

【讨论】:

  • Point 类不可序列化。我不能放弃重写 readObject() 和 writeObject()
  • java.awt.Point 是可序列化的。你用的是其他的吗?
  • 我正在使用导入 android.graphics.Point。此外,当我将此字段设为非静态时,即使使用这些方法,我也会收到异常“Point non serializable”。为什么...?
  • 因为它会尝试自动序列化它们。我会让它们成为非静态和瞬态的。
  • 你能解释一下为什么我不能附加到一个存在的文件吗?我在 fos = new FileOutputStream(file, true); 中使用“true”属性;
【解决方案2】:

@EJP 是对的。它不能附加到使用 ObjectOutputStream 创建的现有文件中。您可以执行以下步骤来修复它: 1.保持ObjectOutputStream对象引用 2. writeObject() 被调用后,不要调用 close() 3.提供关闭ObjectOutputStream的方法。

【讨论】:

  • 问题已经使用自定义ObjectOutputStream解决了
  • 你是怎么解决这个问题的?您介意发布您的自定义 ObjectOutputStream 代码吗?
  • 您可以在问题中找到更新的代码。你只需要重写 writeStreamHeader() 方法
【解决方案3】:

有一种方法可以将多个对象保存在一个文件中:您应该首先创建一个 对象数组(Object [] objects),然后将您的对象一个接一个地作为对象放入该数组中,然后使用writeObject(objects) 方法写入该数组。 祝你好运。

【讨论】:

  • 当您不想将整个数据同时存储在 RAM 中时,此解决方案存在问题。它可以在您保证数据填满 RAM 的情况下工作。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-03-12
  • 2014-09-25
  • 2014-09-25
  • 2020-09-17
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多