2016-04-06 56 views
0

我有一个XML文件,需要基于Xpath变量上的匹配(Contains)进行拆分。使用XSLT将多个文件拆分为xml使用Java

`<Apps> 
    <App> 
     <AppID>49O</AppID> 
     <Type>MP Pay</Type> 
    </App> 
    <App> 
     <AppID>715</AppID> 
     <Type>DP Pay</Type> 
    </App> 
    <App> 
     <AppID>716</AppID> 
     <Type>MP Pay</Type> 
    </App> 
    <App> 
     <AppID>725</AppID> 
     <Type>CP Pay</Type> 
    </App> 
    <App> 
     <AppID>728</AppID> 
     <Type>MP Pay</Type> 
    </App> 
    <App> 
     <AppID>728</AppID> 
     <Type>CP Pay</Type> 
    </App> 
</Apps>` 

所以条件分裂XML是 1.应用程序/应用/类型/文本() - >包含 'MP' 2.应用程序/应用/类型/文本() - >包含 'DP'

以下是在预期输出:

  1. 包含类型为 'MP' 应该是在一个文件中的所有应用程序的节点 - > MP.xml。包含类型为 'DP'

    <Apps> <App> <AppID>49O</AppID> <Type>MP Pay</Type> </App> <App> <AppID>716</AppID> <Type>MP Pay</Type> </App> <App> <AppID>728</AppID> <Type>MP Pay</Type> </App> </Apps>

  2. 所有App节点应该是在一个文件 - > DP.xml。

    <Apps> <App> <AppID>715</AppID> <Type>DP Pay</Type> </App> </Apps>

  3. 所有App节点未能匹配上述2个条件应在文件 - > Mismatched.xml。

    <Apps> <App> <AppID>715</AppID> <Type>CP Pay</Type> </App> <App> <AppID>725</AppID> <Type>CP Pay</Type> </App> </Apps>

的Java代码

public class Splitter { 

    public static void transform(String sourcePath, String xsltPath, 
      String resultDir) { 

     TransformerFactory tFactory = TransformerFactory.newInstance(); 
     try { 
      Transformer transformer = tFactory.newTransformer(new StreamSource(
        new File(xsltPath))); 
      transformer.transform(new StreamSource(new File(sourcePath)), 
        new StreamResult(new File(resultDir))); 
     } catch (Exception e) { 
      e.printStackTrace(); 
     } 
    } 

    public static void main(String[] args) { 

     // Set saxon as transformer. 
     System.setProperty("javax.xml.transform.TransformerFactory", 
       "net.sf.saxon.TransformerFactoryImpl"); 

     String inputFilepath = "resources\\InputFile.xml"; 
     String transformXsltPath = "resources\\Transform.xslt"; 
     String outputDir = "D://Tmp//"; 

     transform(inputFilepath, transformXsltPath, outputDir); 
     System.out.println("Completed"); 
    } 

} 
+0

你可以发布你的java代码呢?你使用撒克逊,普通的Java(与xslt 2.0)? 谢谢, 斯蒂芬 –

+0

是的,我正在使用撒克逊。在我的问题本身中添加了java代码 –

回答

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

    <xsl:template match="Apps"> 
     <xsl:result-document href="MP.xml"> 
     <xsl:copy> 
      <xsl:apply-templates select="App[contains(Type, 'MP')]"/> 
     </xsl:copy> 
     </xsl:result-document> 
     <xsl:result-document href="DP.xml"> 
     <xsl:copy> 
      <xsl:apply-templates select="App[contains(Type, 'DP')]"/> 
     </xsl:copy> 
     </xsl:result-document> 
     <xsl:result-document href="Mismatched.xml"> 
     <xsl:copy> 
      <xsl:apply-templates select="App[not(contains(Type, 'MP') or contains(Type, 'DP'))]"/> 
     </xsl:copy> 
     </xsl:result-document> 
    </xsl:template> 

    <xsl:template match="@* | node()"> 
     <xsl:copy> 
     <xsl:apply-templates select="@* | node()"/> 
     </xsl:copy> 
    </xsl:template> 
</xsl:stylesheet> 
+0

谢谢。完美地工作! –

0

下面是这一种方法更有章可循:

<xsl:template match="App[starts-with(Type, 'MP')]" mode="group">MP</xsl:template> 

<xsl:template match="App[starts-with(Type, 'DP')]" mode="group">DP</xsl:template> 

<xsl:template match="App" mode="group">Mismatched</xsl:template> 

<xsl:function name="f:key" as="xs:string"> 
    <xsl:param name="e" as="element(App)"/> 
    <xsl:apply-templates select="." mode="group"/> 
</xsl:function> 

<xsl:template match="Apps"> 
    <xsl:for-each-group select="App" group-by="f:key(.)"> 
     <xsl:result-document href="{current-grouping-key()}.xml"> 
     <Apps> 
      <xsl:copy-of select="current-group()"/> 
     </Apps> 
     </xsl:result-document> 
    </xsl:for-each-group> 
</xsl:template> 
相关问题