jjzjj

xml - XSLT:我可以创建一个本身就是模板的函数库吗?

coder 2024-06-30 原文

我有几个 XSLT 转换。它们都包含相同的模板,看起来像这样(它的实现和功能无关紧要):

<xsl:template match="firstField| secondField | thirdField">
    <xsl:element name="{local-name(.)}">    
    <xsl:choose>
        <xsl:when test="string-length(.)!=0"><xsl:value-of select="."/></xsl:when>  
        <xsl:otherwise>ABSENT</xsl:otherwise>   
    </xsl:choose>   
    </xsl:element>
</xsl:template>

如您所见,我已经列出了此模板中要应用于模板匹配的字段。

但实际上我想在我所有的转换中使用这个模板,但当然要使用不同的字段名称。换句话说,我想像使用函数一样使用它,我可以插入任何 .xsl 文件并指定参数列表,这些参数是要以这种方式修改的字段的名称。

我可以用 XSLT 做吗?

最佳答案

更新,我可能误解了你的问题并再次查看了你的模板(请参阅下面的解决方案方向的更通用描述)。

你写道:

<xsl:template match="firstField| secondField | thirdField">
    <xsl:element name="{local-name(.)}">    
    <xsl:choose>
        <xsl:when test="string-length(.)!=0"><xsl:value-of select="."/></xsl:when>  
        <xsl:otherwise>ABSENT</xsl:otherwise>   
    </xsl:choose>   
    </xsl:element>
</xsl:template>

和:

I want to use it like a function I can plug in to any .xsl file and specify a list of arguments, which are the names of the fields to be modified this way.

如果用“this way”你的意思是:

  • 将本地名称作为新元素名称(即,我假设您故意选择将其从 namespace 中剥离,否则,xsl:copy 就足够了)
  • 如果元素内容为空,输出“ABSENT”
  • 否则,取当前节点的值

然后您可以按以下方式进行。但这仅在您的要求像您所说的那样普遍适用于您的用例时才有效。

编写模板如下:

<xsl:variable name="names">
    <names>
        <n>firstField</n>
        <n>secondField</n>
        <n>thirdField</n>
    </names>
</xsl:variable>

<xsl:template match="*" mode="by-name">
   <xsl:element name="{local-name(.)}">
      <xsl:apply-templates select="self::node()" mode="text" />
   </xsl:element>
</xsl:template>

<xsl:template match="node()" mode="by-name" />

<xsl:template match="*[text()]" mode="text">
    <xsl:copy />
</xsl:template>

<xsl:template match="*[not(text())]" mode="text">
    <xsl:text>ABSENT</xsl:text>
</xsl:template>

<xsl:template match="/">
    <xsl:apply-templates select="your/current/whatever" />
</xsl:template>

<xsl:template match="*">
    <xsl:for-each select="exslt:node-set($names)/names/n/text()">
        <xsl:apply-templates select="self::*[name() = .]" mode="by-name" />
    </xsl:for-each>
</xsl:template>

将上面的代码放在一个单独的文件中,除了变量,你应该把它放在那里,但留空。使用 xsl:import 导入此文件,您现在唯一需要做的就是覆盖 xsl:variable

在 XSLT 2.0 和 3.0 中有比这更通用和更简单的方法,但这个版本适用于 XSLT 1.0。

免责声明:未经测试,可能包含错误,当然,请根据您的需要进行调整;)

更好/通用的方法

是的,你可以做到这一点。不同的 XSLT 版本有不同的抽象层次:

XSLT 1.0

您可以创建命名模板(只需为其命名)。如果您使用 xsl:call-template 调用命名模板,则当前上下文项将用作模板内的上下文项。这将解决您的模板匹配字段名称的问题。

您可以将它放在一个单独的文件中并使用 xsl:import 导入它,这允许您在必要时覆盖它,或者使用 xsl:include,它不会允许覆盖,如果出现命名冲突,将引发错误。

XSLT 2.0

您可以在 XSLT 2.0 中创建一个可以像调用任何其他函数一样调用的函数。函数可以包含您在上面显示的模板。

在 XSLT 2.0 中,您还可以使用导入和包含。

XSLT 3.0

您可以像在以前的版本中那样做,但您现在可以将它们放在(预编译的)包中,这使得重用、重新分发和调用它们变得更加容易。

此外,XSLT 3.0 在覆盖和接受/公开已用包的组件方面有一个大大改进的系统。

所有版本

您可能有一个地方正在使用 xsl:apply-templates。如果你想防止重复声明匹配的 xsl:template,你可以通过创建一个通用的来解决这个问题:

<xsl:template match="node()" mode="special">
    <xsl:call-template name="yourNamedTemplate" />
</xsl:template>

然后使用以下方法“调用”:

<xsl:apply-templates select="firstField | secondField | thirdField" mode="special" />

提示

如果可重用性很重要,那么您的“库样式表”(在 XSLT 3.0 中会弹出官方术语“库包”),您应该在命名空间中创建模式名称。事实上,可重用样式表中的任何命名组件(命名模板、模式、函数、累加器、键)都应该在它们自己的命名空间中。这可以防止冲突,如果用户想要覆盖它们,他们将必须明确地这样做。

您可以创建一个“继承链”。如果 A 导入 B 导入 C,那么最高优先级被赋予 A 中的命名组件,然后是 B,然后是 C。对于冲突的匹配模板也是如此。这(通常)在您的主要样式表中是不允许的(因此设置了优先级),但是 A 可以具有与 B 或 C 相同的匹配模板。在这种情况下,A 超越 B 超越 C。

关于xml - XSLT:我可以创建一个本身就是模板的函数库吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32585889/

有关xml - XSLT:我可以创建一个本身就是模板的函数库吗?的更多相关文章

  1. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  2. ruby - 如何在 Ruby 中顺序创建 PI - 2

    出于纯粹的兴趣,我很好奇如何按顺序创建PI,而不是在过程结果之后生成数字,而是让数字在过程本身生成时显示。如果是这种情况,那么数字可以自行产生,我可以对以前看到的数字实现垃圾收集,从而创建一个无限系列。结果只是在Pi系列之后每秒生成一个数字。这是我通过互联网筛选的结果:这是流行的计算机友好算法,类机器算法:defarccot(x,unity)xpow=unity/xn=1sign=1sum=0loopdoterm=xpow/nbreakifterm==0sum+=sign*(xpow/n)xpow/=x*xn+=2sign=-signendsumenddefcalc_pi(digits

  3. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  4. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  5. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  6. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  7. ruby-on-rails - 如何从 format.xml 中删除 <hash></hash> - 2

    我有一个对象has_many应呈现为xml的子对象。这不是问题。我的问题是我创建了一个Hash包含此数据,就像解析器需要它一样。但是rails自动将整个文件包含在.........我需要摆脱type="array"和我该如何处理?我没有在文档中找到任何内容。 最佳答案 我遇到了同样的问题;这是我的XML:我在用这个:entries.to_xml将散列数据转换为XML,但这会将条目的数据包装到中所以我修改了:entries.to_xml(root:"Contacts")但这仍然将转换后的XML包装在“联系人”中,将我的XML代码修改为

  8. ruby - 我可以使用 Ruby 从 CSV 中删除列吗? - 2

    查看Ruby的CSV库的文档,我非常确定这是可能且简单的。我只需要使用Ruby删除CSV文件的前三列,但我没有成功运行它。 最佳答案 csv_table=CSV.read(file_path_in,:headers=>true)csv_table.delete("header_name")csv_table.to_csv#=>ThenewCSVinstringformat检查CSV::Table文档:http://ruby-doc.org/stdlib-1.9.2/libdoc/csv/rdoc/CSV/Table.html

  9. ruby-on-rails - 无法使用 Rails 3.2 创建插件? - 2

    我对最新版本的Rails有疑问。我创建了一个新应用程序(railsnewMyProject),但我没有脚本/生成,只有脚本/rails,当我输入ruby./script/railsgeneratepluginmy_plugin"Couldnotfindgeneratorplugin.".你知道如何生成插件模板吗?没有这个命令可以创建插件吗?PS:我正在使用Rails3.2.1和ruby​​1.8.7[universal-darwin11.0] 最佳答案 随着Rails3.2.0的发布,插件生成器已经被移除。查看变更日志here.现在

  10. ruby - 如何使用 RSpec::Core::RakeTask 创建 RSpec Rake 任务? - 2

    如何使用RSpec::Core::RakeTask初始化RSpecRake任务?require'rspec/core/rake_task'RSpec::Core::RakeTask.newdo|t|#whatdoIputinhere?endInitialize函数记录在http://rubydoc.info/github/rspec/rspec-core/RSpec/Core/RakeTask#initialize-instance_method没有很好的记录;它只是说:-(RakeTask)initialize(*args,&task_block)AnewinstanceofRake

随机推荐