我有2个Document对象,文档包含类似的XML。例如:
<tt:root xmlns:tt="http://myurl.com/">
<tt:child/>
<tt:child/>
</tt:root>
另一个:
<ns1:root xmlns:ns1="http://myurl.com/" xmlns:ns2="http://myotherurl.com/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<ns1:child/>
<ns1:child xsi:type="ns2:SomeType"/>
</ns1:root>
我需要将它们合并到包含1个根元素和4个子元素的1个文档中。
问题是,如果我使用document.importNode
函数进行合并,它会正确处理各地的命名空间但是xsi:type元素。所以我得到的结果是:
<tt:root xmlns:tt="http://myurl.com/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<tt:child/>
<tt:child/>
<ns1:child xmlns:ns1="http://myurl.com/"/>
<ns1:child xmlns:ns1="http://myurl.com/" xsi:type="ns2:SomeType"/>
</tt:root>
如您所见,ns2在xsi:type
中使用,但未在任何地方定义。有没有自动解决这个问题的方法?
感谢。
增加:
如果使用默认的java DOM库无法完成此任务,可能还有一些其他库可用于完成我的任务?
答案 0 :(得分:2)
如果我修复了第二个文件中的Namespace问题(通过绑定“xsi”前缀),并使用下面的代码进行合并,则命名空间绑定将保留在输出中;或者至少它们在这里(Windows版本1.6.0_24上的vanilla Java 64位)。
String s1 = "<!-- 1st XML document here -->";
String s2 = "<!-- 2nd XML document here -->";
DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
factory.setNamespaceAware( true );
DocumentBuilder builder = factory.newDocumentBuilder();
Document doc1 = builder.parse( new ByteArrayInputStream( s1.getBytes() ) );
Document doc2 = builder.parse( new ByteArrayInputStream( s2.getBytes() ) );
Element doc1root = ( Element )doc1.getDocumentElement();
Element doc2root = ( Element )doc2.getDocumentElement();
NamedNodeMap atts1 = doc1root.getAttributes();
NamedNodeMap atts2 = doc2root.getAttributes();
for( int i = 0; i < atts1.getLength(); i++ )
{
String name = atts1.item( i ).getNodeName();
if( name.startsWith( "xmlns:" ) )
{
if( atts2.getNamedItem( name ) == null )
{
doc2root.setAttribute( name, atts1.item( i ).getNodeValue() );
}
}
}
NodeList nl = doc1.getDocumentElement().getChildNodes();
for( int i = 0; i < nl.getLength(); i++ )
{
Node n = nl.item( i );
doc2root.appendChild( doc2.importNode( n, true ) );
}
TransformerFactory transformerFactory = TransformerFactory.newInstance();
Transformer transformer = transformerFactory.newTransformer();
StreamResult streamResult = new StreamResult( System.out );
transformer.transform( new DOMSource( doc2 ), streamResult );
答案 1 :(得分:2)
这里的问题是在属性值中使用名称空间前缀;在创建命名空间标准时从未考虑过的东西,以及常见的Java DOM / XML工具无法轻松处理的东西。但是,您可以通过
解决它xsi:type="prefix:value"
的每个实例替换为xsi:type="{namespace}value"
。通过这样做,您不依赖于前缀映射。在您的示例中,<xsi:type="ns2:SomeType"
将变为xsi:type="{http://myotherurl.com/}SomeType"
。答案 2 :(得分:1)
我会使用JAXB和Mergeable plugin在模式派生类中生成mergeFrom
方法。然后:
JAXB通常处理xsi:type
非常好。
答案 3 :(得分:1)
<强>更新强>
这不适用于的情况 这两个文件发生了冲突 名称空间前缀(来自的映射) 第二份文件将取代 从第一个映射。)
您可以将名称空间声明从第二个文档复制到导入的节点。由于子节点可以覆盖父节点前缀,因此这是有效的:
<foo:root xmlns:foo="urn:ROOT">
<foo:child xmlns:foo="urn:CHILD" xsi:type="foo:child-type">
...
</foo:child>
</foo:root>
在上面的XML中,绑定到前缀“foo”的名称空间在子元素的范围内被覆盖。您可以通过执行以下操作来完成此用例:
import java.io.File;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import org.w3c.dom.Attr;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
public class Demo {
public static void main(String[] args) throws Exception {
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(true);
DocumentBuilder db = dbf.newDocumentBuilder();
File file1 = new File("src/forum231/input1.xml");
Document doc1 = db.parse(file1);
Element rootElement1 = doc1.getDocumentElement();
File file2 = new File("src/forum231/input2.xml");
Document doc2 = db.parse(file2);
Element rootElement2 = doc2.getDocumentElement();
// Copy Child Nodes
NodeList childNodes2 = rootElement2.getChildNodes();
for(int x=0; x<childNodes2.getLength(); x++) {
Node importedNode = doc1.importNode(childNodes2.item(x), true);
if(importedNode.getNodeType() == Node.ELEMENT_NODE) {
Element importedElement = (Element) importedNode;
// Copy Attributes
NamedNodeMap namedNodeMap2 = rootElement2.getAttributes();
for(int y=0; y<namedNodeMap2.getLength(); y++) {
Attr importedAttr = (Attr) doc1.importNode(namedNodeMap2.item(y), true);
importedElement.setAttributeNodeNS(importedAttr);
}
}
rootElement1.appendChild(importedNode);
}
// Output Document
TransformerFactory tf = TransformerFactory.newInstance();
Transformer t = tf.newTransformer();
DOMSource source = new DOMSource(doc1);
StreamResult result = new StreamResult(System.out);
t.transform(source, result);
}
}
<强>输出强>
<?xml version="1.0" encoding="UTF-8" standalone="no"?><tt:root xmlns:tt="http://myurl.com/">
<tt:child/>
<tt:child/>
<ns1:child xmlns:ns1="http://myurl.com/" xmlns:ns2="http://myotherurl.com/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"/>
<ns1:child xmlns:ns1="http://myurl.com/" xmlns:ns2="http://myotherurl.com/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:type="ns2:SomeType"/>
</tt:root>
原始回答
除了复制元素外,您还可以复制属性。这将确保生成的文档包含必要的名称空间声明:
import java.io.File;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import org.w3c.dom.Attr;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
public class Demo {
public static void main(String[] args) throws Exception {
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(true);
DocumentBuilder db = dbf.newDocumentBuilder();
File file1 = new File("input1.xml");
Document doc1 = db.parse(file1);
Element rootElement1 = doc1.getDocumentElement();
File file2 = new File("input2.xml");
Document doc2 = db.parse(file2);
Element rootElement2 = doc2.getDocumentElement();
// Copy Attributes
NamedNodeMap namedNodeMap2 = rootElement2.getAttributes();
for(int x=0; x<namedNodeMap2.getLength(); x++) {
Attr importedNode = (Attr) doc1.importNode(namedNodeMap2.item(x), true);
rootElement1.setAttributeNodeNS(importedNode);
}
// Copy Child Nodes
NodeList childNodes2 = rootElement2.getChildNodes();
for(int x=0; x<childNodes2.getLength(); x++) {
Node importedNode = doc1.importNode(childNodes2.item(x), true);
rootElement1.appendChild(importedNode);
}
// Output Document
TransformerFactory tf = TransformerFactory.newInstance();
Transformer t = tf.newTransformer();
DOMSource source = new DOMSource(doc1);
StreamResult result = new StreamResult(System.out);
t.transform(source, result);
}
}
<强>输出:强>
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<tt:root xmlns:tt="http://myurl.com/" xmlns:ns1="http://myurl.com/" xmlns:ns2="http://myotherurl.com/" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<tt:child/>
<tt:child/>
<ns1:child/>
<ns1:child xsi:type="ns2:SomeType"/>
</tt:root>
答案 4 :(得分:1)
单行XQuery可以完成这项任务:构建一个名为上下文根元素的新节点,然后将其子节点与其他文档中的子节点一起导入:
declare variable $other external; element {node-name(*)} {*/*, $other/*/*}
虽然在XQuery中你没有完全控制命名空间节点(至少在XQuery 1.0中),但是它有一个copy-namespaces
模式设置,可以用来要求保持命名空间上下文不变,以防万一实现确实默认保留它。
如果XQuery是一个可行的选项,那么saxon9he.jar可能是您所追求的“神奇的xml库”。
以下是使用s9api API:
公开某些上下文的示例代码import javax.xml.parsers.DocumentBuilderFactory;
import net.sf.saxon.s9api.*;
import org.w3c.dom.Document;
...
Document merge(Document context, Document other) throws Exception
{
Processor processor = new Processor(false);
XQueryExecutable executable = processor.newXQueryCompiler().compile(
"declare variable $other external; element {node-name(*)} {*/*, $other/*/*}");
XQueryEvaluator evaluator = executable.load();
DocumentBuilder db = processor.newDocumentBuilder();
evaluator.setContextItem(db.wrap(context));
evaluator.setExternalVariable(new QName("other"), db.wrap(other));
Document doc =
DocumentBuilderFactory.newInstance().newDocumentBuilder().newDocument();
processor.writeXdmValue(evaluator.evaluate(), new DOMDestination(doc));
return doc;
}
答案 5 :(得分:0)
如果您知道要添加的名称空间URI和前缀URI,则可以像简单地向元素添加属性一样简单。当我的合并文档缺少xmlns时,这对我有用:xsd =“http://www.w3.org/2001/XMLSchema”包含在我导入的文档中:
myDocument.getDocumentElement.setAttribute("xmlns:xsd", "http://www.w3.org/2001/XMLSchema");