Java List在序列化的时候调用ObjectOutputStream.writeObject出现java.lang.StackOverflowError异常的解决

    最近项目遇到了一个bug,是一个树状结构的数据太多太深导致的java.lang.StackOverflowError的bug。

    项目是Android项目,起因是项目需要保存全局的机构树,但是由于Android Application可能出现被回收导致空指针异常,因此数据除了在全局的Application保存一份之外,还将数据备份在磁盘上,如果Application出现异常数据被清掉,那么就从磁盘上读取数据,这个逻辑是没有问题的。现在关键是数据的序列化及反序列化保存,之前的数据存储是先通过ObjectOutputStream序列化生成byte数组,然后在sqlite3中存储,如下:

  //保存
  public void saveObject(Object object, String column) {
        ByteArrayOutputStream arrayOutputStream = new ByteArrayOutputStream();
        try {
            ObjectOutputStream objectOutputStream = new ObjectOutputStream(arrayOutputStream);
            objectOutputStream.writeObject(object);
            objectOutputStream.flush();
            byte data[] = arrayOutputStream.toByteArray();
            objectOutputStream.close();
            arrayOutputStream.close();
            mDB.execSQL("UPDATE " + TABLE_TMP +  " SET " + column + "  = ?", new Object[] { data });
        } catch (Exception e) {
            e.printStackTrace();
        }
    }


    //获取
    public Object getObject(String column) {
        Object object = null;
        Cursor cursor = mDB.rawQuery("select * from " + TABLE_TMP , null);
        if (cursor != null) {
            if (cursor.moveToFirst()) {
                byte data[] = cursor.getBlob(cursor.getColumnIndex(column));
                try {
                    ByteArrayInputStream arrayInputStream = new ByteArrayInputStream(data);
                    ObjectInputStream inputStream = new ObjectInputStream(arrayInputStream);
                    object = inputStream.readObject();
                    inputStream.close();
                    arrayInputStream.close();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
            cursor.close();
        }
        return object;
    }

    这种代码大多数情况下都没有问题,可是当存储的Object变成ArrayList,并且ArrayList为树状的嵌套数据的时候,如果嵌套太深,就会报标题出现的错误。先看看栈溢出的定义:如果一个线程在计算时所需要用到栈大小 > 配置允许最大的栈大小,那么Java虚拟机将抛出StackOverflowError,因此,最容易出现栈溢出的情况是递归太深。本例中,就是由于ArrayList的嵌套太深导致系统所需的栈大小超出最大栈大小。实际上,本例的这个bug,在某些比较老的手机上会出现,而在一些配置比较好的新手机上不会出现,原因是由于Android设备的不断更新,硬件设备性能的提示,系统配置参数dalvik相关属性也在提升。

    为了修复这个bug,本文采用的方式是将ArrayList转为String进行存储,避免递归调用,然后读取的时候再将其转换回ArrayList,代码如下:

    /**
    *保存列表形式的数据,原来采用saveObject,但是由于List的列表是树状机构,可能存在太深的情况
     * 而这种情况会导致ObjectOutputStream出现Stack Overflow的bug,为了避免这个bug,将数据转换为字符串存储
    **/
    public <T> void saveList(List<T> list, String column) {
        Gson gson = new Gson();
        String inputString= gson.toJson(list);
        saveObject(inputString, column);
    }

    /**
     *获取列表形式的数据,将数据转换读取为字符串,然后再转换成list
     **/
    public <T> List<T> getList(String column) {
        Gson gson = new Gson();
        Type type = new TypeToken<List<T>>() {}.getType();
        String s = (String) getObject(column);
        if(s != null){
            return gson.fromJson(s, type);
        }
        return  null;
    }

    其中saveObject和getObject参考最上面的代码。

Java中,`ObjectOutputStream.writeObject()`方法用于将对象序列化为字节流,以便进行持久化存储或网络传输。如果需要将`Blob`类型的数据作为参数传递给该方法,需确保`Blob`数据能够被序列化Java本身并不直接支持`Blob`类型的序列化,因此需要将`Blob`转换为可序列化的类型(如字节数组`byte[]`),或者确保`Blob`对象本身实现了`Serializable`接口。 ### 使用字节数组存储Blob数据 由于`Blob`本质上是一组二进制数据,最常见的方式是将其读取为字节数组`byte[]`,然后通过`ObjectOutputStream`写入流中。以下是一个示例: ```java import java.io.*; import java.sql.Blob; import java.sql.SQLException; public class BlobSerializationExample { public static void serializeBlob(Blob blob, String filePath) throws IOException, SQLException { // 将Blob转换为字节数组 byte[] blobBytes = blob.getBytes(1, (int) blob.length()); try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(filePath))) { oos.writeObject(blobBytes); // 写入字节数组 } } public static Blob deserializeBlob(String filePath) throws IOException, ClassNotFoundException { try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream(filePath))) { byte[] blobBytes = (byte[]) ois.readObject(); // 读取字节数组 return new javax.sql.rowset.serial.SerialBlob(blobBytes); // 转换为Blob } } } ``` 在该示例中,`Blob`对象被转换为`byte[]`后,通过`ObjectOutputStream.writeObject()`方法序列化。反序列化时,使用`SerialBlob`类将字节数组重新包装为`Blob`对象[^3]。 ### 实现Serializable接口的Blob封装类 若需要直接传递`Blob`对象,可创建一个封装类并实现`Serializable`接口: ```java import java.io.Serializable; import java.sql.Blob; public class SerializableBlob implements Serializable { private static final long serialVersionUID = 1L; private final byte[] data; public SerializableBlob(Blob blob) throws SQLException { this.data = blob.getBytes(1, (int) blob.length()); } public Blob toBlob() { return new javax.sql.rowset.serial.SerialBlob(data); } } ``` 使用该类进行序列化和反序列化: ```java public static void serializeBlob(SerializableBlob serializableBlob, String filePath) throws IOException { try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream(filePath))) { oos.writeObject(serializableBlob); // 写入封装的Blob对象 } } public static SerializableBlob deserializeBlob(String filePath) throws IOException, ClassNotFoundException { try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream(filePath))) { return (SerializableBlob) ois.readObject(); // 读取封装的Blob对象 } } ``` 通过封装类,`Blob`数据可以被正确序列化并作为参数传递给`ObjectOutputStream.writeObject()`方法[^1]。 ### 注意事项 - `Blob`本身不是`Serializable`接口的实现类,因此不能直接通过`ObjectOutputStream.writeObject()`进行序列化。 - 若`Blob`对象未正确转换为字节数组或未封装为可序列化的类,调用`writeObject()`时会抛出`NotSerializableException`异常。 - 在处理大体积的`Blob`数据时,应考虑内存使用情况,避免因字节数组过大导致内存溢出。 ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值