如何在XML中拆分字符串

如何在XML中拆分字符串,xml,string,xslt,split,Xml,String,Xslt,Split,我有这种XSL <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0"> <xsl:output indent="yes"/> <xsl:template match="dataroot"> <xml><xsl:apply-templates/></xml> </xsl:te

我有这种XSL

<xsl:stylesheet  xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output indent="yes"/>

    <xsl:template match="dataroot">
        <xml><xsl:apply-templates/></xml>
    </xsl:template>

    <xsl:template match="M_17">
        <package id="{package_id}" cat="{cat}">
            <nazwa><xsl:value-of select="nazwa"/></nazwa>
            <xsl:if test="author"><author><xsl:value-of select="author"/></author></xsl:if>
            <xsl:if test="www"><www><xsl:value-of select="translate(www,'#','')"/></www></xsl:if>
            <xsl:if test="opis"><opis><xsl:value-of select="opis"/></opis></xsl:if>
            <xsl:if test="img"><img><xsl:value-of select="translate(img,'#','')"/></img></xsl:if>

            <xsl:if test="depends"><depends><xsl:value-of select="depends"/></depends></xsl:if>
            <xsl:if test="conflicts"><conflicts><xsl:value-of select="conflicts"/></conflicts></xsl:if>
            <xsl:if test="after"><after><xsl:value-of select="after"/></after></xsl:if>
            <xsl:if test="replaces"><replaces><xsl:value-of select="replaces"/></replaces></xsl:if>
        </package>
    </xsl:template>

</xsl:stylesheet>

但是,当在eg中有2个值时,这两个值将在代码中显示

<depends>modload com1node</depends>
modload-com1node
但我想通过XSL将其转换为:

<depends>modloader</depends>
<depends>com1node</depends>
modloader
COM1节点
这应该发生在:依赖、冲突、之后和替换

如何将这些字符串(如果它们出现在源XML中)拆分为简单的字符串(如我在示例中所示,每行一个)

核心XML的一部分

<?xml version="1.0" encoding="UTF-8"?>
<dataroot xmlns:od="urn:schemas-microsoft-com:officedata" generated="2014-05-11T15:51:32">
    <Mnc_172>
        <ID>1</ID>
        <package_id>minecraft</package_id>
        <cat>lib</cat>
        <www>#http://minecraft.net/#</www>
        <nazwa>Minecraft</nazwa>
        <author>Mojang</author>
        <opis>Game - build your own world!</opis>
        <img>#/mc.png#</img>
    </Mnc_172>
    <Mnc_172>
        <ID>2</ID>
        <package_id>modloader</package_id>
        <cat>lib</cat>
        <www>#http://minecraftforum.net/topic/75440-x/#</www>
        <nazwa>ModLoader</nazwa>
        <author>Risugami</author>
        <opis>ModLoader - library to load mods</opis>
        <img>#/gen.png#</img>
        <replaces>modL forging</replaces>
    </Mnc_172>
    ...
</dataroot>

1.
我的世界
解放党
#http://minecraft.net/#
我的世界
莫江
游戏-建立你自己的世界!
#/mc.png#
2.
modloader
解放党
#http://minecraftforum.net/topic/75440-x/#
ModLoader
里苏加米
ModLoader-用于加载mods的库
#/巴布亚新几内亚#
模锻
...

XML与XSLT:
MU17
Mnc\U 172
不匹配。无论如何,在XSLT1.0中,您需要使用递归模板来标记内容。因此,尝试改变:

<depends><xsl:value-of select="depends"/></depends>

致:


并将以下模板添加到样式表中:

<xsl:template name="tokenize">
    <xsl:param name="text"/>
    <xsl:param name="elemName"/>
    <xsl:param name="sep" select="' '"/>
    <xsl:choose>
        <xsl:when test="contains($text, $sep)">
            <xsl:element name="{$elemName}">
                <xsl:value-of select="substring-before($text, $sep)"/>
            </xsl:element>
            <!-- recursive call -->
            <xsl:call-template name="tokenize">
                <xsl:with-param name="text" select="substring-after($text, $sep)" />
                <xsl:with-param name="elemName" select="$elemName" />
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:element name="{$elemName}">
                <xsl:value-of select="$text"/>
            </xsl:element>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>

XML与XSLT:
MU17
Mnc\U 172
不匹配。无论如何,在XSLT1.0中,您需要使用递归模板来标记内容。因此,尝试改变:

<depends><xsl:value-of select="depends"/></depends>

致:


并将以下模板添加到样式表中:

<xsl:template name="tokenize">
    <xsl:param name="text"/>
    <xsl:param name="elemName"/>
    <xsl:param name="sep" select="' '"/>
    <xsl:choose>
        <xsl:when test="contains($text, $sep)">
            <xsl:element name="{$elemName}">
                <xsl:value-of select="substring-before($text, $sep)"/>
            </xsl:element>
            <!-- recursive call -->
            <xsl:call-template name="tokenize">
                <xsl:with-param name="text" select="substring-after($text, $sep)" />
                <xsl:with-param name="elemName" select="$elemName" />
            </xsl:call-template>
        </xsl:when>
        <xsl:otherwise>
            <xsl:element name="{$elemName}">
                <xsl:value-of select="$text"/>
            </xsl:element>
        </xsl:otherwise>
    </xsl:choose>
</xsl:template>

XSLT2.0有一个简单的函数来标记字符串,但在XSLT1.0中,您必须更具创造性。我通常会用一个递归模板来攻击这样的东西,该模板对第一个空格之前的文本进行处理,然后对剩余的文本进行递归调用,当文本用完时停止

<xsl:stylesheet  xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output indent="yes"/>

    <xsl:template match="dataroot">
        <xml><xsl:apply-templates/></xml>
    </xsl:template>

    <xsl:template match="M_17">
        <package id="{package_id}" cat="{cat}">
            <nazwa><xsl:value-of select="nazwa"/></nazwa>
            <xsl:if test="author"><author><xsl:value-of select="author"/></author></xsl:if>
            <xsl:if test="www"><www><xsl:value-of select="translate(www,'#','')"/></www></xsl:if>
            <xsl:if test="opis"><opis><xsl:value-of select="opis"/></opis></xsl:if>
            <xsl:if test="img"><img><xsl:value-of select="translate(img,'#','')"/></img></xsl:if>

            <xsl:apply-templates select="depends | conflicts | after | replaces" />
        </package>
    </xsl:template>

    <xsl:template match="depends | conflicts | after | replaces">
        <xsl:param name="text" select="concat(normalize-space(), ' ')" />
        <xsl:if test="$text">
            <xsl:copy>
                <xsl:value-of select="substring-before($text, ' ')" />
            </xsl:copy>
            <xsl:apply-templates select=".">
                <xsl:with-param name="text" select="substring-after($text, ' ')" />
            </xsl:apply-templates>
        </xsl:if>
    </xsl:template>
</xsl:stylesheet>
将复制输入元素范围内的命名空间声明。这是无害的,但你可以认为它看起来有点乱。为了避免这种情况,您可以使用

<xsl:element name="{local-name()}">
    <xsl:value-of select="substring-before($text, ' ')" />
</xsl:element>


相反。

XSLT2.0有一个简单的函数来标记字符串,但在XSLT1.0中,您必须更具创造性。我通常会用一个递归模板来攻击这样的东西,该模板对第一个空格之前的文本进行处理,然后对剩余的文本进行递归调用,当文本用完时停止

<xsl:stylesheet  xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
    <xsl:output indent="yes"/>

    <xsl:template match="dataroot">
        <xml><xsl:apply-templates/></xml>
    </xsl:template>

    <xsl:template match="M_17">
        <package id="{package_id}" cat="{cat}">
            <nazwa><xsl:value-of select="nazwa"/></nazwa>
            <xsl:if test="author"><author><xsl:value-of select="author"/></author></xsl:if>
            <xsl:if test="www"><www><xsl:value-of select="translate(www,'#','')"/></www></xsl:if>
            <xsl:if test="opis"><opis><xsl:value-of select="opis"/></opis></xsl:if>
            <xsl:if test="img"><img><xsl:value-of select="translate(img,'#','')"/></img></xsl:if>

            <xsl:apply-templates select="depends | conflicts | after | replaces" />
        </package>
    </xsl:template>

    <xsl:template match="depends | conflicts | after | replaces">
        <xsl:param name="text" select="concat(normalize-space(), ' ')" />
        <xsl:if test="$text">
            <xsl:copy>
                <xsl:value-of select="substring-before($text, ' ')" />
            </xsl:copy>
            <xsl:apply-templates select=".">
                <xsl:with-param name="text" select="substring-after($text, ' ')" />
            </xsl:apply-templates>
        </xsl:if>
    </xsl:template>
</xsl:stylesheet>
将复制输入元素范围内的命名空间声明。这是无害的,但你可以认为它看起来有点乱。为了避免这种情况,您可以使用

<xsl:element name="{local-name()}">
    <xsl:value-of select="substring-before($text, ' ')" />
</xsl:element>


相反。

请参见中的
fn:tokenize
我对这类东西一无所知。这个密码不是我的。我在5小时前见过XSL,所以当我还不完全理解一些代码组合时,您能告诉我如何在代码中插入XSL。请您发布一个XML源代码的示例吗?@Admaster,对不起,我一直在使用XQuery和XPath,它们使用相同的函数,以便我可以为您指出正确的一个,但是我不知道/不做XSLT。@CharlesDuffy您的建议需要XSLT2.0;OP的样式表显示XSLT1.0。请参阅中的
fn:tokenize
,我在这类内容中是空白的。这个密码不是我的。我在5小时前见过XSL,所以当我还不完全理解一些代码组合时,您能告诉我如何在代码中插入XSL。请您发布一个XML源代码的示例吗?@Admaster,对不起,我一直在使用XQuery和XPath,它们使用相同的函数,以便我可以为您指出正确的一个,但是我不知道/不做XSLT。@CharlesDuffy您的建议需要XSLT2.0;OP的样式表显示XSLT 1.0。在before track中有一个错误,原因是此代码添加了类似于w3.org/2001/XMLSchema-instance的smth">modL@Admaster这本身不是一个错误,只是
从源元素复制作用域名称空间的方式。由于这些名称空间没有在输出元素中使用,因此它们不会对语义造成任何影响,但是如果出于美观的原因想删除它们,那么只需使用
..
而不是
..
。在before track中有一个错误,因为此代码添加了类似于w3.org/2001/XMLSchema-instance的smth">modL@Admaster这本身不是一个错误,只是
从源元素复制作用域名称空间的方式。由于这些名称空间没有在输出元素中使用,因此它们对语义没有任何影响,但是如果出于美观的原因想删除它们,那么只需使用
..
而不是
..
。我知道这种“错误”。我在数据库中有4个完全相同的表。代码工作正常,但我不明白为什么有$elemName和{}@Admaster一起使用和不使用{}@Admaster一起搜索属性值模板。我在数据库中有4个完全相同的表。代码工作正常,但不理解为什么有$elemName,有和没有{}@Admaster,请搜索属性值模板。