使用 XMLEncoder 的 Java Bean 持久性

标签 java persistence javabeans

我编写了一个包含 HashMultiMap 的 bean 类(来自 Guava 库)。我想使用 JRE 的 XMLEncoder 对 bean 进行 XML 编码。使用自定义 PersistenceDelegate,我已成功将 bean 写入文件。但是,当我尝试反序列化 XML 时出现异常:

java.lang.NoSuchMethodException: <unbound>=HashMultimap.put("pz1", "pz2")

我做错了什么?

// create the bean
SomeBean sb = new SomeBean();

// add some data
HashMultimap<String, String> stateMap = HashMultimap.create();    
stateMap.put("pz1", "pz2");
stateMap.put("pz3", "pz4");

sb.setStateMap(stateMap);

// encode as xml 
FileOutputStream os = new FileOutputStream("myXMLFile.xml");
XMLEncoder encoder = new XMLEncoder(os);
encoder.setPersistenceDelegate(HashMultimap.class, new CustomPersistenceDelegate());
encoder.writeObject(sb);

// decode the xml
FileInputStream is = new FileInputStream("myXMLFile.xml"); 
XMLDecoder decoder = new XMLDecoder(is);
Object deSerializedObject = decoder.readObject();

class CustomPersistenceDelegate extends DefaultPersistenceDelegate
{
    protected Expression instantiate(Object oldInstance, Encoder out)
    {
        return new Expression(oldInstance, oldInstance.getClass(), "create", null);
    }

    protected void initialize(Class<?> type, Object oldInstance, Object newInstance,
                              Encoder out)
    {
        super.initialize(type, oldInstance, newInstance, out);

        com.google.common.collect.HashMultimap<String, String> m =
            (com.google.common.collect.HashMultimap) oldInstance;

        for (Map.Entry<String, String> entry : m.entries())
        {
            out.writeStatement(new Statement(oldInstance, "put", 
            new Object[] { entry.getKey(), entry.getValue() }));
        }

    }
}

public class SomeBean
{
    private HashMultimap<String, String> stateMap;

    public HashMultimap<String, String> getStateMap()
    {
        return stateMap;
    }

    public void setStateMap(HashMultimap<String, String> stateMap)
    {
        this.stateMap = stateMap;
    }
}

最佳答案

我(还)没有解决方案。但这里至少可以澄清问题。似乎在 Java 7 build 15 及更高版本中所做的一些更改破坏了您的 Statement 所需的方法查找。如果您将 ExceptionListener 添加到 XmlEncoder,它可以让您更好地了解这是如何失败的:

        encoder.setExceptionListener(new ExceptionListener() {
        @Override
        public void exceptionThrown(Exception e) {
            System.out.println("got exception. e=" + e);
            e.printStackTrace();
        }
    });

然后您将看到完整的堆栈跟踪:

java.lang.Exception: Encoder: discarding statement HashMultimap.put(Object, Object);
at java.beans.Encoder.writeStatement(Encoder.java:306)
at java.beans.XMLEncoder.writeStatement(XMLEncoder.java:400)
at test2.XmlEncoderTest$CustomPersistenceDelegate.initialize(XmlEncoderTest.java:83)
at java.beans.PersistenceDelegate.writeObject(PersistenceDelegate.java:118)
at java.beans.Encoder.writeObject(Encoder.java:74)
at java.beans.XMLEncoder.writeObject(XMLEncoder.java:327)
at java.beans.Encoder.writeExpression(Encoder.java:330)
at java.beans.XMLEncoder.writeExpression(XMLEncoder.java:454)
at java.beans.PersistenceDelegate.writeObject(PersistenceDelegate.java:115)
at java.beans.Encoder.writeObject(Encoder.java:74)
at java.beans.XMLEncoder.writeObject(XMLEncoder.java:327)
at java.beans.Encoder.writeExpression(Encoder.java:330)
at java.beans.XMLEncoder.writeExpression(XMLEncoder.java:454)
at java.beans.DefaultPersistenceDelegate.doProperty(DefaultPersistenceDelegate.java:194)
at java.beans.DefaultPersistenceDelegate.initBean(DefaultPersistenceDelegate.java:253)
at java.beans.DefaultPersistenceDelegate.initialize(DefaultPersistenceDelegate.java:400)
at java.beans.PersistenceDelegate.writeObject(PersistenceDelegate.java:118)
at java.beans.Encoder.writeObject(Encoder.java:74)
at java.beans.XMLEncoder.writeObject(XMLEncoder.java:327)
at java.beans.Encoder.writeExpression(Encoder.java:330)
at java.beans.XMLEncoder.writeExpression(XMLEncoder.java:454)
at java.beans.PersistenceDelegate.writeObject(PersistenceDelegate.java:115)
at java.beans.Encoder.writeObject(Encoder.java:74)
at java.beans.XMLEncoder.writeObject(XMLEncoder.java:327)
at java.beans.Encoder.writeObject1(Encoder.java:258)
at java.beans.Encoder.cloneStatement(Encoder.java:271)
at java.beans.Encoder.writeStatement(Encoder.java:301)
at java.beans.XMLEncoder.writeStatement(XMLEncoder.java:400)
at java.beans.XMLEncoder.writeObject(XMLEncoder.java:330)

    ...

Caused by: java.lang.NoSuchMethodException: HashMultimap.put(Object, Object);
    at java.beans.Statement.invokeInternal(Statement.java:313)
    at java.beans.Statement.access$000(Statement.java:58)
    at java.beans.Statement$2.run(Statement.java:185)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.beans.Statement.invoke(Statement.java:182)
    at java.beans.Statement.execute(Statement.java:173)
    at java.beans.Encoder.writeStatement(Encoder.java:304)
    ... 51 more

Caused by 部分显示它未能找到 put 方法。在我看来,发生这种情况是因为它无法再正确匹配方法签名。它在 java bean MethodFinder 中失败,但由于源代码未包含在 JDK 中,我无法很好地找到它。

如果我能找到确切的原因,我会更新这个。只是想同时为您提供更多信息。

更新

我认为这是这些更高版本中的错误。这是一个更直接地暴露错误(或意外行为)的单元测试。下面的失败正是您的代码中发生的事情:

@Test
public void testMethodFinder() throws Exception {

    Method m0 = MethodFinder.findMethod(this.getClass(), "setUp", new Class<?>[0]);
    assertNotNull(m0);

    // this is okay, because method is declared in the type referenced
    Method m = MethodFinder.findMethod(Multimap.class, "put", new Class<?>[] { Object.class, Object.class });
    assertNotNull(m);

    try {
        // this fails, apparently because method is not declared in this subclass (is inherited from parent class)
        Method m2 = MethodFinder.findMethod(HashMultimap.class, "put", new Class<?>[] { Object.class, Object.class });
        assertNotNull(m2);
    } catch (Exception e) {
        System.out.println("got exception. e=" + e);
    }
}

关于使用 XMLEncoder 的 Java Bean 持久性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15888383/

相关文章:

clojure - 在 Clojure 中保存映射/结构的最简单方法是什么?

redis - 重启 Redis 副本

java - 不可变业务对象和 MessagePack 消息之间的自动转换

java - NoSuchBeanDefinition异常 : No qualifying bean of type 'org.springframework.mail.MailSender'

java - 使用 Marathon 运行 Spark 作业

java - WIldfly - 身份验证成功后的sql查询

java - JBoss EAP 6.1 错误 : This runtime type requires a JDK. 此服务器执行环境的当前默认 VM 未被识别为 JDK

iphone - iOS 中持久数据应该存储在哪里?

java - 转发到相同的 struts2 操作时丢失 bean 的值

java - 在 Spring Integration 中为 ContentEnricher 定义异常回退行为