将XML元素拆分为多个
这可能是不可能的,但你们可能有答案, 我正在尝试拆分此xml将XML元素拆分为多个,xml,xslt,tokenize,Xml,Xslt,Tokenize,这可能是不可能的,但你们可能有答案, 我正在尝试拆分此xml <CTP> <name>ABSA bank</name> <BAs.BA>bank|sector|issuer</BAs.BA> </CTP> 阿布萨银行 银行|部门|发行人 并将其转换为以下形式: <CTP> <name>ABSA bank</name> <BAs>
<CTP>
<name>ABSA bank</name>
<BAs.BA>bank|sector|issuer</BAs.BA>
</CTP>
阿布萨银行
银行|部门|发行人
并将其转换为以下形式:
<CTP>
<name>ABSA bank</name>
<BAs>
<BA>bank</BA>
<BA>sector</BA>
<BA>issuer</BA>
</BAs>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs.FA>dep|sec|issue</BAs.BA>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs>
<FA>bank</BA>
<FA>sector</BA>
<FA>issuer</BA>
</FAs>
</CTP>
阿布萨银行
银行
部门
发行人
我可以使用此代码执行此操作
<?xml version="1.0"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:str="http://exslt.org/strings" version="1.0" extension-element-prefixes="str">
<xsl:template name="str:tokenize">
<xsl:param name="string" select="''" />
<xsl:param name="delimiters" select="'	
'" />
<xsl:choose>
<xsl:when test="not($string) or (string-length($string)=0)" />
<xsl:when test="not($delimiters) or (string-length($delimiters)=0)">
<xsl:call-template name="str:_tokenize-characters">
<xsl:with-param name="string" select="$string" />
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="str:_tokenize-delimiters">
<xsl:with-param name="string" select="$string" />
<xsl:with-param name="delimiters" select="$delimiters" />
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template name="str:_tokenize-characters">
<xsl:param name="string" />
<xsl:if test="$string">
<BA>
<xsl:value-of select="substring($string, 1, 1)" />
</BA>
<xsl:call-template name="str:_tokenize-characters">
<xsl:with-param name="string" select="substring($string, 2)" />
</xsl:call-template>
</xsl:if>
</xsl:template>
<xsl:template name="str:_tokenize-delimiters">
<xsl:param name="string" />
<xsl:param name="delimiters" />
<xsl:variable name="delimiter" select="substring($delimiters, 1, 1)" />
<xsl:choose>
<xsl:when test="string-length($delimiter)=0">
<BA>
<xsl:value-of select="$string" />
</BA>
</xsl:when>
<xsl:when test="contains($string, $delimiter)">
<xsl:choose>
<xsl:when test="not(starts-with($string, $delimiter))">
<xsl:call-template name="str:_tokenize-delimiters">
<xsl:with-param name="string" select="substring-before($string, $delimiter)" />
<xsl:with-param name="delimiters" select="substring($delimiters, 2)" />
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<BA />
</xsl:otherwise>
</xsl:choose>
<xsl:call-template name="str:_tokenize-delimiters">
<xsl:with-param name="string" select="substring-after($string, $delimiter)" />
<xsl:with-param name="delimiters" select="$delimiters" />
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="str:_tokenize-delimiters">
<xsl:with-param name="string" select="$string" />
<xsl:with-param name="delimiters" select="substring($delimiters, 2)" />
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template match="/">
<xsl:variable name="v1">
<xsl:value-of select="CTP/BAs.BA" />
</xsl:variable>
<BAs>
<xsl:call-template name="str:tokenize">
<xsl:with-param name="string" select="$v1" />
<xsl:with-param name="delimiters" select="'#'" />
</xsl:call-template>
</BAs>
</xsl:template>
</xsl:stylesheet>
但主要的挑战是让它充满活力,换句话说,我想要“BAs”
以及动态写入“BA”节点,以防得到其他XML,如:
<CTP>
<name>ABSA bank</name>
<BAs>
<BA>bank</BA>
<BA>sector</BA>
<BA>issuer</BA>
</BAs>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs.FA>dep|sec|issue</BAs.BA>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs>
<FA>bank</BA>
<FA>sector</BA>
<FA>issuer</BA>
</FAs>
</CTP>
阿布萨银行
副证券发行
我希望它看起来像这样:
<CTP>
<name>ABSA bank</name>
<BAs>
<BA>bank</BA>
<BA>sector</BA>
<BA>issuer</BA>
</BAs>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs.FA>dep|sec|issue</BAs.BA>
</CTP>
<CTP>
<name>ABSA bank</name>
<FAs>
<FA>bank</BA>
<FA>sector</BA>
<FA>issuer</BA>
</FAs>
</CTP>
阿布萨银行
银行
部门
发行人
试着这样做:
XSLT1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<!-- identity transform -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="*[contains(name(), '.')]" name="nest">
<xsl:param name="name" select="name()"/>
<xsl:param name="delimiter" select="'.'"/>
<xsl:choose>
<xsl:when test="contains($name, $delimiter)">
<xsl:element name="{substring-before($name, $delimiter)}" >
<!-- recursive call -->
<xsl:call-template name="nest">
<xsl:with-param name="name" select="substring-after($name, $delimiter)"/>
</xsl:call-template>
</xsl:element>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="tokenize">
<xsl:with-param name="name" select="$name"/>
<xsl:with-param name="text" select="."/>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<xsl:template name="tokenize">
<xsl:param name="name"/>
<xsl:param name="text"/>
<xsl:param name="delimiter" select="'|'"/>
<xsl:variable name="token" select="substring-before(concat($text, $delimiter), $delimiter)" />
<xsl:if test="$token">
<xsl:element name="{$name}" >
<xsl:value-of select="$token"/>
</xsl:element>
</xsl:if>
<xsl:if test="contains($text, $delimiter)">
<!-- recursive call -->
<xsl:call-template name="tokenize">
<xsl:with-param name="name" select="$name"/>
<xsl:with-param name="text" select="substring-after($text, $delimiter)"/>
</xsl:call-template>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
应用于以下测试输入:
XML
<CTP>
<name>ABSA bank</name>
<BAs.BA>bank|sector|issuer</BAs.BA>
<FAs.FA.F>dep|sec|issue</FAs.FA.F>
</CTP>
阿布萨银行
银行|部门|发行人
副证券发行
结果将是:
<?xml version="1.0" encoding="UTF-8"?>
<CTP>
<name>ABSA bank</name>
<BAs>
<BA>bank</BA>
<BA>sector</BA>
<BA>issuer</BA>
</BAs>
<FAs>
<FA>
<F>dep</F>
<F>sec</F>
<F>issue</F>
</FA>
</FAs>
</CTP>
阿布萨银行
银行
部门
发行人
副署长
秒
问题
这个问题模棱两可。元素名称中是否总是有两个令牌,或者可以有更多的令牌?作为第一步,我将使用两个令牌进行尝试,如果有效,我们将使用2+令牌进行尝试,但首先我的目标是实现2令牌挑战@迈克尔。hor257k@michael.hor257k,你能检查一下第二步吗?你认为它可行吗?我相信这是可能的,但我没有时间讨论到这样的细节。我相信你现在有了自己的工具来解决这个问题。我会尽我最大的努力。非常感谢。