Java 中的高效 XSLT 管道(或将结果重定向到源)

2022-09-02 09:56:43

我有一系列XSL 2.0样式表,它们相互馈送,即样式表A的输出馈送B馈送C。

最有效的方法是什么?重新表述的问题是:如何有效地将一种转换的输出路由到另一种转换中。

这是我的第一次尝试:

@Override
public void transform(Source data, Result out) throws TransformerException{
    for(Transformer autobot : autobots){
        if(autobots.indexOf(autobot) != (autobots.size()-1)){
            log.debug("Transforming prelim stylesheet...");
            data = transform(autobot,data);
        }else{
            log.debug("Transforming final stylesheet...");
            autobot.transform(data, out);
        }
    }
}

private Source transform(Transformer autobot, Source data) throws TransformerException{
    DOMResult result = new DOMResult();
    autobot.transform(data, result);
    Node node = result.getNode();
    return new DOMSource(node);
}

如您所见,我正在使用DOM来坐在转换之间,尽管它很方便,但它在性能方面不是最佳的。

有没有简单的方法可以路由说,路由SAXResult到SAXSource?StAX解决方案将是另一种选择。

我知道像XProc这样的项目,如果你还没有看过,这是非常酷的,但我不想投资整个框架。


答案 1

我发现这个:#3。链接转换,显示使用 TransformerFactory 链接转换的两种方法,让一个转换的结果馈送下一个转换,然后最终输出到系统输出。这避免了在转换之间对字符串、文件等进行中间序列化的需要。

当需要对同一 XML 文档进行多次连续转换时,请确保避免不必要的分析操作。我经常遇到将一个字符串转换为另一个字符串,然后将该字符串转换为另一个字符串的代码。这不仅很慢,而且还会消耗大量内存,尤其是在不允许对中间字符串进行垃圾回收的情况下。

大多数转换都基于一系列 SAX 事件。SAX 解析器通常会将 InputStream 或其他 InputSource 解析为 SAX 事件,然后将其馈送到 Transformer。与其将转换器输出到文件、字符串或其他此类结果,不如使用 SAXResult。SAXResult接受一个ContentHandler,它可以将这些SAX事件直接传递给另一个Transformer,依此类推。

这是一种方法,也是我通常更喜欢的方法,因为它为各种输入和输出源提供了更大的灵活性。它还使得动态创建具有可变数量转换的转换链变得相当容易。

SAXTransformerFactory stf = (SAXTransformerFactory)TransformerFactory.newInstance();

// These templates objects could be reused and obtained from elsewhere.
Templates templates1 = stf.newTemplates(new StreamSource(
  getClass().getResourceAsStream("MyStylesheet1.xslt")));
Templates templates2 = stf.newTemplates(new StreamSource(
  getClass().getResourceAsStream("MyStylesheet1.xslt")));

TransformerHandler th1 = stf.newTransformerHandler(templates1);
TransformerHandler th2 = stf.newTransformerHandler(templates2);

th1.setResult(new SAXResult(th2));
th2.setResult(new StreamResult(System.out));

Transformer t = stf.newTransformer();
t.transform(new StreamSource(System.in), new SAXResult(th1));

// th1 feeds th2, which in turn feeds System.out.

答案 2

相关问题 高效 XSLT 管道,带有参数,在 Java 中阐明了传递给此类转换器链的正确参数。

它还暗示了没有第三个变压器的稍短解决方案:

SAXTransformerFactory stf = (SAXTransformerFactory)TransformerFactory.newInstance();

Templates templates1 = stf.newTemplates(new StreamSource(
        getClass().getResourceAsStream("MyStylesheet1.xslt")));
Templates templates2 = stf.newTemplates(new StreamSource(
        getClass().getResourceAsStream("MyStylesheet2.xslt")));

TransformerHandler th1 = stf.newTransformerHandler(templates1);
TransformerHandler th2 = stf.newTransformerHandler(templates2);

th2.setResult(new StreamResult(System.out));

// Note that indent, etc should be applied to the last transformer in chain:
th2.getTransformer().setOutputProperty(OutputKeys.INDENT, "yes");

th1.getTransformer().transform(new StreamSource(System.in), new SAXResult(th2));