2

我必须使用 commons-digester.jar 在 android 中处理 xml 文件实际上这是一个开源程序,它使用 commons-digester.jar 在 Java 中进行 xml 处理,我需要更改它以支持 Android,但是会发生此错误:

Digester.getParser:java.lang.UnsupportedOperationException:此解析器不支持规范“未知”版本“0.0”

java.lang.NullPointerException 03-29 11:24:02.590: W/System.err(17018): at org.apache.commons.digester3.Digester.getXMLReader(Digester.java:790) 03-29 11:24:02.590 : W/System.err(17018): at org.apache.commons.digester3.Digester.parse(Digester.java:1588) 03-29 11:24:02.590: W/System.err(17018): at org. apache.commons.digester3.Digester.parse(Digester.java:1557) 03-29 11:24:02.590: W/System.err(17018): at com.tashkeel.android.utilities.alkhalil.DbLoader.LoadPrefixes(DbLoader .java:65) 03-29 11:24:02.590: W/System.err(17018): 在 com.tashkeel.android.utilities.alkhalil.analysis.Analyzer.(Analyzer.java:64) 03-29 11: 24:02.600: W/System.err(17018): 在 com.tashkeel.android.MainActivity$1.run(MainActivity.java:80) 03-29 11:24:02.600: W/System.err(17018): 在java.lang.Thread.run(Thread.java:856) 03-29 11:24:03.240: W/System.err(17018):

和使用 commons-digester 的代码示例

 Digester digester = new Digester();
 digester.addObjectCreate("prefixes", Lists.class);
 digester.addObjectCreate("prefixes/prefixe", Prefixe.class);
  digester.addSetProperties("prefixes/prefixe", "unvoweledform",     
           "unvoweledform");
 digester.addSetProperties("prefixes/prefixe", "voweledform",

  "voweledform");
  digester.addSetProperties("prefixes/prefixe", "desc", "desc");

 digester.addSetProperties("prefixes/prefixe", "classe", "classe");

 digester.addSetNext("prefixes/prefixe", "addPrefixe");

 return (Lists)digester.parse(pref);

我尝试解析的 xml 的一部分:

  <?xml version="1.0" encoding="utf-8" ?>
<prefixes>
    <prefixe unvoweledform="" voweledform="" desc="" classe="C1">
    </prefixe>
    <prefixe unvoweledform="و" voweledform="وَ" desc="حرف العطف" classe="C1">
        </prefixe>
    <prefixe unvoweledform="ف" voweledform="فَ" desc="حرف العطف أو الاستئناف" 
classe="C1">
     </prefixe>
    </prefixes>
4

2 回答 2

1

查看 Digester 的源代码,归结为以下代码:

SAXParserFactory factory = SAXParserFactory.newInstance();
factory.setNamespaceAware( namespaceAware );
factory.setXIncludeAware( xincludeAware );
factory.setValidating( validating );
factory.setSchema( schema );

然后检查SAXParserFactory#setSchema的 JavaDoc :

解析器必须能够使用任何 Schema 实现... 抛出:UnsupportedOperationException - 当实现不覆盖此方法时。

所以我的猜测是SAXParserFactory您的代码正在接收的实现在某种程度上不支持模式。

SAXParserFactory.html#newInstance文档中有一些故障排除技巧,但您也可以通过以下内容找到返回的确切解析器类(不需要 Digester 代码):

SAXParserFactory factory = SAXParserFactory.newInstance();
System.err.println("SAXParserFactory class is " + factory.getClass().getName());

此外,来自Android SAXParserFactory doco

UnsupportedOperationException 为了向后兼容,当使用 JAXP 早期版本的实现时,将抛出此异常。

因此,您可能需要检查您正在使用的 JAXP 等版本。

更新

鉴于您不需要模式,您可以尝试这个可能有效的方法......而不是让消化器SAXParserFactory为您创建,自己创建一个 - 这样您就可以避免调用setSchema看起来可能是问题的方法:

// create your own SAXParserFactory, but don't call any of the set methods unless you explicitly need to
final SAXParserFactory myfactory = SAXParserFactory.newInstance();

//myfactory.setNamespaceAware( namespaceAware );
//myfactory.setXIncludeAware( xincludeAware );
//myfactory.setValidating( validating );
//myfactory.setSchema( schema );

// create your own Digester and override the getFactory method to return your own factory
Digester digester = new Digester() {
  @Override
  public SAXParserFactory getFactory() {
    return myfactory;
  }
};

// use this digester as normal for the rest of your code...

目前没有进行任何android开发,所以无法测试,但看起来它可以工作......

更新 2

所以看起来它解决了架构问题,而您现在所拥有的似乎是一个单独的类加载器问题。这实际上是一个单独的问题,但这里有一个可能的黑客......绕过banutils内省,只需使用自定义规则自己处理属性。这只是真正实用的,因为您的示例中有相当简单的 XML - 任何比这更复杂的东西,并且这种方法变得不可行:

final List<Prefixe> list = new ArrayList<>();
final SAXParserFactory factory = SAXParserFactory.newInstance();

Digester digester = new Digester() {
  @Override
  public SAXParserFactory getFactory() {
     System.out.println("using custom factory...");
    return factory;
  }
};

digester.addRule("prefixes/prefixe", new Rule() {
  @Override
  public void begin(String namespace, String name, Attributes attributes) throws Exception {
    Prefixe prefixe = new Prefixe();
    prefixe.setUnvoweledform(attributes.getValue("unvoweledform"));
    prefixe.setVoweledform(attributes.getValue("voweledform"));
    prefixe.setDesc(attributes.getValue("desc"));
    prefixe.setClasse(attributes.getValue("classe"));
    list.add(prefixe);
  }
});

digester.parse(...);

请注意,您似乎在某个地方遇到了类加载器问题(除非您使用的 beanutils 版本在某种程度上被削弱了?)我强烈建议您尝试深入了解这些问题,因为它们可能会回来咬您一次又一次...

祝你好运!

于 2015-04-04T01:27:55.703 回答
0

在对@Barney 答案稍作修改后的最终答案,以使从函数返回的列表在 addRule 函数中添加前缀:

    public Lists LoadPrefixes() throws Exception {

    InputStream   in = assetManager.open("db/prefixes.xml");

   final SAXParserFactory factory = SAXParserFactory.newInstance();

    Digester digester = new Digester() {
      @Override
      public SAXParserFactory getFactory() {
         //System.out.println("using custom factory...");
        return factory;
      }
    };

      final Lists Lists_ob=new Lists();


    digester.addRule("prefixes/prefixe", new Rule() {
      @Override
      public void begin(String namespace, String name, Attributes attributes) throws Exception {
        Prefixe prefixe = new Prefixe();
        prefixe.setUnvoweledform(attributes.getValue("unvoweledform"));
        prefixe.setVoweledform(attributes.getValue("voweledform"));
        prefixe.setDesc(attributes.getValue("desc"));
        prefixe.setClasse(attributes.getValue("classe"));
        Lists_ob.addPrefixe(prefixe);

      }
    });


      digester.parse(in);

     return Lists_ob;

}

这是代码最初在 Lists.class 中

 private List prefixes = new LinkedList();

   public void addPrefixe(Prefixe p) {
    prefixes.add(p);
}
于 2015-04-06T22:38:07.490 回答