Сериализация / десериализация ClassCastException: x не может быть приведен к java.io.ObjectStreamClass

Используя собственную сериализацию Java, я периодически вижу ClassCastExceptions

java.lang.ClassCastException: myCompany.MyClass$MembershipServiceMethod cannot be cast to java.io.ObjectStreamClass

или (реже)

java.lang.ClassCastException: java.lang.String cannot be cast to java.io.ObjectStreamClass

когда я десериализую объекты определенного неизменного класса. То есть, исключение всегда выдается для определенных сериализованных представлений, но большинство объектов могут быть успешно сериализованы и десериализованы.

public final class ServiceInteractionImpl implements ServiceInteraction, Serializable {
private static final long serialVersionUID = 1L;

private final InteractionSource source;
private final long startTime;
private final InteractionType type;
private final ServiceMethod method;
private final long duration;
private final String accompanyingMessage;

public ServiceInteractionImpl(final ServiceMethod method,
                                  final InteractionSource source,
                                  final InteractionType type,
                                  final long startTime,
                                  final long duration,
                                  final String accompanyingMessage) {
            this.source = source;
            this.startTime = startTime;
            this.duration = duration;
            this.type = type;
            this.method = method;
            this.accompanyingMessage = accompanyingMessage;
    }

    ...

    getters and canonical methods
}

где InteractionSource, InteractionType и ServiceMethod являются перечислениями. Я не могу воспроизвести это в моей локальной среде, где выполняются тесты junit, которые сериализуют и десериализуют миллионы объектов.

Вот код написания

    fileLock.lock();
    try {
        final File recordFile = new File(recordFileName);
        boolean appendToFile = false;

        if (recordFile.exists()) {
            appendToFile = true;
        }

        final OutputStream fileOutputStream = new FileOutputStream(recordFileName, true);
        final OutputStream buffer = new BufferedOutputStream(fileOutputStream);

        ObjectOutput output;
        if (appendToFile) {
            output = new AppendableObjectOutputStream(buffer);
        } else {
            output = new ObjectOutputStream(buffer);
        }

        int localSerializedInteractionsCount = 0;

        try {
            for (final ServiceInteraction interaction : tempCollection) {
                output.writeObject(interaction);
                output.flush();
                localSerializedInteractionsCount++;
                rollBackCollection.remove(interaction);
            }
        } finally {
            output.close();
            serializedInteractionCount += localSerializedInteractionsCount;
        }
    } catch (IOException e) {
        LOGGER.error("could not write to file", e);
        //some roll-back code
    } finally {
        fileLock.unlock();
    }

см. Добавление к ObjectOutputStream для AppendableObjectOutputStream

Любая помощь высоко ценится.

1 ответ

Если определение класса изменилось без обновления serialVersionUID, то возможно, что эти ошибки происходят из устаревших экземпляров объекта на основе старого определения класса.

Другие вопросы по тегам