Apache Digester:NoSuchMethodException:没有这样的可访问方法

时间:2014-08-08 12:28:07

标签: java xml parsing xml-parsing apache-commons-digester

我遇到Apache Digester的问题,因为它在当前对象上调用 set method 而不是在与父标记对应的对象中调用 。结果是NoSuchMethodException。

Note: I'm reusing the Digester object due to performance problems, but I'm calling clear() method before each parse.

以下是我的代码:( 在标题对象上调用setHeader 方法而不是消息对象)。

XML:

<message>
    <header id_message="2014871" date="07/08/2014 21:56:41" />
    <body>
        ...
    </body>
</message>

创建消化器的代码:

digesterSOBB = new Digester();
digesterSOBB.setValidating(true);
digesterSOBB.setSchema(XMLParser.class.getResource(dtdUrlSOBB).getFile());
digesterSOBB.register("message", XMLParser.class.getResource(dtdUrlSOBB));
digesterSOBB.setErrorHandler(new XMLDtdValidationErrorHandler());

//Message
digesterSOBB.addObjectCreate("message", "com.company.utils.xmlparser.beans.sosc.Message");
digesterSOBB.addSetProperties("message");

//Header-I
digesterSOBB.addObjectCreate("message/header", "com.company.utils.xmlparser.beans.sosc.Header"); 
digesterSOBB.addSetProperties("message/header");
digesterSOBB.addSetNext("message/header", "setHeader", "com.company.utils.xmlparser.beans.sosc.Header");

它工作正常,但是当它处理一些xmls时,它会尝试在Header对象中调用setHeader方法:

[8/7/14 5:22:00:322 CEST] 00000061 Digester      E org.apache.commons.logging.impl.Jdk14Logger error End event threw exception
java.lang.NoSuchMethodException: No such accessible method: setHeader() on object: com.company.utils.xmlparser.Header
at org.apache.commons.beanutils.MethodUtils.invokeMethod(MethodUtils.java:278)
at org.apache.commons.digester.SetNextRule.end(SetNextRule.java:217)
at org.apache.commons.digester.Rule.end(Rule.java:253)
at org.apache.commons.digester.Digester.endElement(Digester.java:1332)
at org.apache.xerces.parsers.AbstractSAXParser.endElement(Unknown Source)
at org.apache.xerces.parsers.AbstractXMLDocumentParser.emptyElement(Unknown Source)
...

我查看了javadoc(http://commons.apache.org/proper/commons-digester/commons-digester-2.0/docs/api/),代码似乎没问题(第二个参数是在父元素上调用的方法名称):

public void addSetNext(String pattern,
                       String methodName,
                       String paramType)
Add a "set next" rule for the specified parameters.
Parameters:
pattern - Element matching pattern
methodName - Method name to call on the parent element
paramType - Java class name of the expected parameter type (if you wish to use a primitive type, specify the corresonding Java wrapper class instead, such as java.lang.Boolean for a boolean parameter)

1 个答案:

答案 0 :(得分:1)

我没有使用过这个库,但是查看手册,它是基于堆栈的 - 引用父级(Message)时是否存在排序问题,可能是在子级(Header)下,较新的对象位于顶部堆栈?

常见问题解答中有一些信息: http://wiki.apache.org/commons/Digester/FAQ

参见 - &gt; 如何在SetNextRule之前触发CallMethodRule? 建议的示例序列是:

digester.addObjectCreate("parent", Parent.class);
digester.addObjectCreate("parent/child", Child.class);
digester.addSetNext("parent/child", "addChild");
digester.addCallMethod("parent/child", "setName", 1);
digester.addCallParam("parent/child/name", 0);

希望这有帮助。