35

如何在运行时从序列化过程中排除类字段?编译时间有瞬态修饰符,但运行时呢?我的意思是使用 ObjectOutputStream 进行常见的 java 序列化,而不是 gson 之类的。

对不起,我认为我没有解释正确。这不完全是关于序列化,而是关于序列化。我有一批遗留文件并像这样处理它们:

public class Deserialize {

/**
 * @param args
 * @throws IOException 
 * @throws ClassNotFoundException 
 */
public static void main(String[] args) throws ClassNotFoundException, IOException {
    File file = new File("/home/developer/workspace/DDFS/some.ddf");
    HackedObjectInputStream in = new HackedObjectInputStream(new GZIPInputStream(new FileInputStream(file)));

    System.out.println("Attempt to open " + file.getAbsolutePath());
    Object obj = in.readObject();
    in.close();


}

 static class HackedObjectInputStream extends ObjectInputStream
    {

        /**
         * Migration table. Holds old to new classes representation.
         */
        private static final Map<String, Class<?>> MIGRATION_MAP = new HashMap<String, Class<?>>();

        static
        {
            MIGRATION_MAP.put("DBOBExit", Exit.class);
        }

        /**
         * Constructor.
         * @param stream input stream
         * @throws IOException if io error
         */
        public HackedObjectInputStream(final InputStream stream) throws IOException
        {
            super(stream);
        }

        @Override
        protected ObjectStreamClass readClassDescriptor() throws IOException, ClassNotFoundException
        {
            ObjectStreamClass resultClassDescriptor = super.readClassDescriptor();

            for (final String oldName : MIGRATION_MAP.keySet())
            {
                if (resultClassDescriptor.getName().equals(oldName))
                {
                    resultClassDescriptor = ObjectStreamClass.lookup(MIGRATION_MAP.get(oldName));   
                }
            }

            return resultClassDescriptor;
        }

    }

}

此代码适用于大多数文件,但有些文件会抛出

Exception in thread "main" java.lang.ClassCastException: cannot assign instance of java.awt.Polygon to field Exit.msgbackPt of type java.awt.Point in instance of Exit
at java.io.ObjectStreamClass$FieldReflector.setObjFieldValues(ObjectStreamClass.java:2053)

因为 Exit 类的版本不同。新版本有新领域。当我将瞬态添加到新字段时错误消失,但另一个文件开始抛出异常(最新文件)。

那么,如果我检测到遗留的序列化文件,我可以在运行时向这些新文件添加瞬态吗?也许反射或什么?

4

4 回答 4

60

ObjectOutputStream的文档说:

对象的默认序列化机制写入对象的类、类签名以及所有非瞬态和非静态字段的值。对其他对象的引用(瞬态或静态字段除外)也会导致这些对象被写入。

因此,当您将变量声明为瞬态时,ObjectOutputStream 应将其忽略。确保使用transient关键字而不是@Transient注释。一些 ORM 框架使用此类注释来标记不应保存在数据库中的字段。它们对于内置序列化框架毫无意义。

private transient String foo; // Field gets ignored by ObjectOutputStream
@Transient private String bar; // Treated normally by ObjectOutputStream (might mean something for some other framework)
于 2013-01-29T12:09:39.303 回答
3

您可以使用transient修饰符:

http://docs.oracle.com/javase/specs/jls/se7/html/jls-8.html#jls-8.3.1.3

于 2013-01-30T01:12:50.483 回答
0

特定类的序列化表示取决于类本身,您不能在外部更改它,最接近的方法是定义具有自定义序列化行为的子类,但这只会影响该子类的对象,而不影响父类的对象类型。

如果您根本无法修改有问题的类,那么您唯一的选择是继承 ObjectOutputStream 并覆盖replaceObject以在写入时将问题对象替换为仅包含所需数据的不同对象,并在读取时进行镜像处理(子类 ObjectInputStream 和覆盖resolveObject)。

于 2013-01-29T12:11:18.533 回答
0

你在这里挖错洞了。与其在运行时决定要序列化和重写哪些字段,不如readClassDescriptor()考虑重写readResolve()

于 2013-01-30T00:50:30.773 回答