4

java.lang.NullPointerException当我取消注释以下代码中的一行时,我得到了:

private static Document XSLT(String name) throws Exception{
    File xsl = new File("data/" + name + ".xsl");
    File xml = new File("data/" + name + ".xml");
    DocumentBuilder db = DocumentBuilderFactory.newInstance().newDocumentBuilder();
    Document document = db.newDocument();
    DOMSource input = new DOMSource(db.parse(xml));
    StreamResult output = new StreamResult(System.out); //That works but its not what I want!
    //DOMResult output = new DOMResult(document); //that gives the null pointer exception
    TransformerFactory.newInstance().newTransformer(new StreamSource(xsl)).transform(input, output);
    return document;
}

我的目的是制作一个类似的方法来帮助我转换 XML 文件,同时使用 XSLT 和 Java 语言。毕竟我应该在将其保存到文件之前进行所有转换。

如何从该方法中获取文档?

我的意思是,我想在不将转换后的文档保存到文件中以便能够加载它的情况下执行此操作,我只想在“内存”中执行此操作。

谢谢 :)

更新

让我发布异常,以便您可以更好地帮助我。

Exception in thread "main" javax.xml.transform.TransformerException: java.lang.NullPointerException
        at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:736)
        at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:340)
        at test.Main.XSLT(Main.java:393)
        at test.Main.test(Main.java:342)
        at test.Main.main(Main.java:329)
Caused by: java.lang.NullPointerException
        at com.sun.org.apache.xml.internal.serializer.SerializerBase.patchName(SerializerBase.java:271)
        at com.sun.org.apache.xml.internal.serializer.SerializerBase.addAttribute(SerializerBase.java:429)
        at com.sun.org.apache.xml.internal.serializer.ToSAXHandler.addUniqueAttribute(ToSAXHandler.java:438)
        at xmlname.template$dot$0()
        at xmlname.applyTemplates()
        at xmlname.transform()
        at com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet.transform(AbstractTranslet.java:611)
        at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:729)
        ... 4 more
---------
java.lang.NullPointerException
        at com.sun.org.apache.xml.internal.serializer.SerializerBase.patchName(SerializerBase.java:271)
        at com.sun.org.apache.xml.internal.serializer.SerializerBase.addAttribute(SerializerBase.java:429)
        at com.sun.org.apache.xml.internal.serializer.ToSAXHandler.addUniqueAttribute(ToSAXHandler.java:438)
        at xmlname.template$dot$0()
        at xmlname.applyTemplates()
        at xmlname.transform()
        at com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet.transform(AbstractTranslet.java:611)
        at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:729)
        at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:340)
        at test.Main.XSLT(Main.java:393)
        at test.Main.test(Main.java:342)
        at test.Main.main(Main.java:329)

更新 2

让我们用一个共同的来源进行测试。

测试.xml

<?xml version="1.0" encoding="UTF-8"?>
<tests>
    <test>
        <text>Hi!</text>
    </test>
</tests>

测试.xsl

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

    <xsl:output 
        method="html"
        version="1.0" 
        encoding="utf-8" 
        indent="yes" 
        omit-xml-declaration="no"
        media-type="application/xml"
        doctype-public="-//W3C//DTD XHTML 1.0 Strict//EN" 
        doctype-system="http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"
    />

    <xsl:template match="/">
        <html xmlns="http://www.w3.org/1999/xhtml" lang="pt-BR" xml:lang="pt-BR">           
            <head>
                <meta charset="UTF-8" />
                <title>Test</title>
            </head>
            <body>
                <h1>Test</h1>                       
                <xsl:apply-templates/>
            </body>
        </html> 
    </xsl:template>

    <xsl:template match="/tests/test">
       <p><xsl:value-of select="text"/></p>
    </xsl:template>

</xsl:stylesheet>

使用 给出以下输出StreamResult

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html lang="pt-BR" xml:lang="pt-BR" xmlns="http://www.w3.org/1999/xhtml">
<head>
<META http-equiv="Content-Type" content="text/html; charset=utf-8">
<meta charset="UTF-8">
<title>Test</title>
</head>
<body>
<h1>Test</h1>

<p>Hi!</p>

</body>
</html>

解决了

在适当的地方查看我的答案(在答案中!)

4

3 回答 3

5
DOMResult output = new DOMResult();
TransformerFactory.newInstance().
                   newTransformer(new StreamSource(xsl)).
                   transform(input, output);
return (Document) output.getNode();

参考:
DOMResult#getNode()

获取将包含结果 DOM 树的节点。如果没有通过 DOMResult(Node node), ... 设置节点,则该节点将通过转换设置,并且可以在转换完成后从该方法中获取。

于 2013-07-05T20:20:09.370 回答
1

我怀疑转换的输出格式不正确XML。我可以使用以下代码和输入文件获得相同的异常。

xmlTransform.java

import javax.xml.parsers.*;
import javax.xml.transform.*;
import javax.xml.transform.stream.*;
import javax.xml.transform.dom.*;
import java.io.*;
import org.w3c.dom.*;

public class XmlTransform {
    public static void main(String[] args) throws Exception {
        File xsl = new File("1.xsl");
        File xml = new File("1.xml");
        DocumentBuilder db = DocumentBuilderFactory.newInstance().newDocumentBuilder();
        Document document = db.newDocument();
        DOMSource input = new DOMSource(db.parse(xml));

//        StreamResult output = new StreamResult(System.out); //That works but its not what I want!
        DOMResult output = new DOMResult(document); //that gives the null pointer exception
        TransformerFactory.newInstance().newTransformer(new StreamSource(xsl)).transform(input, output);

        System.out.println("getNode().getLocalName()=" + output.getNode().getLocalName());
        System.out.println("((Document)getNode()).getDocumentElement().getLocalName()=" + ((Document)output.getNode()).getDocumentElement().getLocalName());

        Node n = ((Document)output.getNode()).getDocumentElement();

        System.out.println("n.hasChildNodes()=" + n.hasChildNodes());
        System.out.println("Beatles=" + n.getChildNodes().getLength());
    }
}

1.xml

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" exclude-result-prefixes="xsl">
<xsl:output omit-xml-declaration="yes" indent="yes"/>

<xsl:template match="/">
    <xsl:apply-templates select="//beatle" />
</xsl:template>

<xsl:template match="beatle">
  <xsl:value-of select="name/first"/>
</xsl:template>
</xsl:stylesheet>

1.xsl

<?xml version="1.0" encoding="UTF-8"?>
<beatles>
  <beatle>
    <name>
      <first>John</first>
      <last>Lennon</last>
    </name>
  </beatle>
  <beatle>
    <name>
      <first>Paul</first>
      <last>McCartney</last>
    </name>
  </beatle>
  <beatle>
    <name>
      <first>George</first>
      <last>Harrison</last>
    </name>
  </beatle>
  <beatle>
    <name>
      <first>Ringo</first>
      <last>Starr</last>
    </name>
  </beatle>
</beatles>

编译并运行它,我得到以下异常:

getNode().getLocalName()=null
    Exception in thread "main" java.lang.NullPointerException
    at XmlTransform.main(XmlTransform.java:27)

这与您的不完全相同(见下文!)但替换输出 System.out,我得到

JohnPaulGeorgeRingo

正如预期的那样。

XML现在,如果您开始向 中添加元素,则输出并不完全如此XSLT,例如,让每个名称用 a 包装<name></name>

<name><xsl:value-of select="name/first"/></name>

并切换回DOMResult outputthis 会导致不同的异常:

ERROR: 'HIERARCHY_REQUEST_ERR: An attempt was made to insert a node where it is not permitted. '
    Exception in thread "main" javax.xml.transform.TransformerException: org.w3c.dom.DOMException: HIERARCHY_REQUEST_ERR: An attempt was made to insert a node where it is not permitted.
at com.sun.org.apache.xalan.internal.xsltc.trax.TransformerImpl.transform(TransformerImpl.java:736)

因为输出不是格式良好的XML文档。XML 文档中必须只有一个根节点。所以通过替换:

    <xsl:apply-templates select="//beatle" />

和:

    <names><xsl:apply-templates select="//beatle" /></names>

使输出文档格式正确,最终结果应为:

getNode().getLocalName()=null
((Document)getNode()).getDocumentElement().getLocalName()=names
n.hasChildNodes()=true
Beatles=4

希望这可以帮助。

于 2013-07-05T21:30:12.327 回答
1

XSL文件中的 XHTML是正确的(与 XML 兼容),但 XSLT 处理后的输出不正确。

我查看了这一行(在 XSLT proc 的输出中):

<META http-equiv="Content-Type" content="text/html; charset=utf-8">

themeta全部大写,而 the>应该是/>.

所以我改变了我们看到methodfrom的那一行html

<xsl:output 
    method="html"
    version="1.0" 
    encoding="utf-8" 
    indent="yes" 
    omit-xml-declaration="no"
    media-type="application/xml"
    doctype-public="-//W3C//DTD XHTML 1.0 Strict//EN" 
    doctype-system="http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"
/>

xml

<xsl:output 
    method="xml"
    version="1.0" 
    encoding="utf-8" 
    indent="yes" 
    omit-xml-declaration="no"
    media-type="application/xml"
    doctype-public="-//W3C//DTD XHTML 1.0 Strict//EN" 
    doctype-system="http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"
/>

更新

我遇到了自关闭标签的问题,然后我不得不更改另一件事,以使输出真正符合浏览器的 XHTML。

更改以下行可修复这些问题:

    media-type="application/xhtml+xml"

所以一个空的div:

    <div/>

会写成:

    <div></div>

我使用默认 JDK 附带的 XSLT 处理器对其进行了测试,它可以正常工作。

于 2013-07-05T22:31:21.150 回答