1

我一直在 JMeter 中运行一系列的 web 服务测试,我想做的是取每个测试返回的平均响应时间的总和。我有办法找到每个测试的平均响应时间,但无法将测试的平均值加在一起。我知道,为了使用 XPath 的 sum() 函数,这些值需要成为节点集的一部分,但是据我所知,一旦我从 XML 中找到平均值,它们就不再是其中的一部分。所以我需要使用 node-set() 函数,但我对 XSLT/XPath 还很陌生,并且不确定如何让事情正常工作。

对此的任何帮助表示赞赏!

这是来自 JMeter 的示例 XML,运行两次迭代:

<?xml version="1.0" encoding="UTF-8"?>
<?xml-stylesheet type="text/xsl" href="../style/jmeter-results-detail-report_21.xsl"?>
<testResults version="1.2">
<httpSample t="78" lt="78" ts="1338826079163" s="true" lb="html" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="4418" ng="1" na="1"/>
<httpSample t="31" lt="31" ts="1338826079241" s="true" lb="userRoleRetriever" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="758" ng="1" na="1"/>
<httpSample t="32" lt="32" ts="1338826079272" s="true" lb="UserActivityWSDL" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="2398" ng="1" na="1"/>
<httpSample t="156" lt="125" ts="1338826079304" s="true" lb="SubscriberMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="56434" ng="1" na="1"/>
<httpSample t="31" lt="16" ts="1338826079460" s="true" lb="NetworkMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="33020" ng="1" na="1"/>
<httpSample t="15" lt="15" ts="1338826079507" s="true" lb="AlarmMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="11594" ng="1" na="1"/>
<httpSample t="141" lt="141" ts="1338826079538" s="true" lb="getSubscribers" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="397" ng="1" na="1"/>
<httpSample t="265" lt="234" ts="1338826079679" s="true" lb="getMpegResultsById" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="832927" ng="1" na="1"/>
<httpSample t="15" lt="15" ts="1338826079976" s="true" lb="getOverallSummary" rc="200" rm="OK" tn="vuserver 1-1" dt="text" by="402" ng="1" na="1"/>
<httpSample t="0" lt="0" ts="1338826082663" s="true" lb="html" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="4418" ng="1" na="1"/>
<httpSample t="16" lt="16" ts="1338826082663" s="true" lb="userRoleRetriever" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="758" ng="1" na="1"/>
<httpSample t="15" lt="0" ts="1338826082679" s="true" lb="UserActivityWSDL" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="2398" ng="1" na="1"/>
<httpSample t="32" lt="0" ts="1338826082694" s="true" lb="SubscriberMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="56434" ng="1" na="1"/>
<httpSample t="31" lt="15" ts="1338826082726" s="true" lb="NetworkMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="33020" ng="1" na="1"/>
<httpSample t="16" lt="16" ts="1338826082757" s="true" lb="AlarmMgmtWSDL" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="11594" ng="1" na="1"/>
<httpSample t="250" lt="250" ts="1338826082788" s="true" lb="getSubscribers" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="10536" ng="1" na="1"/>
<httpSample t="15454" lt="15392" ts="1338826083038" s="true" lb="getMpegResultsById" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="2023426" ng="1" na="1"/>
<httpSample t="15" lt="15" ts="1338826098555" s="true" lb="getOverallSummary" rc="200" rm="OK" tn="vuserver 1-2" dt="text" by="402" ng="1" na="1"/>

</testResults>

这是我的 XSLT 中的一个摘录,其中找到了每个测试的平均值。

<xsl:for-each select="/testResults/*[not(@lb = preceding::*/@lb)]">
    ...
    <xsl:variable name="count" select="count(../*[@lb = current()/@lb])" />
    <xsl:variable name="totalTime" select="sum(../*[@lb = current()/@lb]/@t)" />
    <xsl:variable name="averageTime" select="$totalTime div $count" />
    ...
</xsl:for-each>

如何将这些平均时间输入节点集中并随后求和?

以下是感兴趣的人的最终报告的示例

提前致谢!

4

1 回答 1

1

如果您可以使用 XSLT2.0,那么它具有处理节点集的内置功能,而在 XSLT1.0 中您必须使用扩展功能。通过节点集,您可以创建一个变量,其中包含(新创建的)节点列表,然后您可以迭代它们,甚至对它们求和,就好像它们是源文档本身一样。

在展示解决方案之前要注意的另一件事是,这也是分组问题的一个示例。您有多个测试的结果,并希望按名称对它们进行分组。您当前使用的方法不一定那么有效。在 XSLT2.0 中虽然有xsl:for-each-group函数使事情变得更容易

<xsl:for-each-group select="httpSample" group-by="@lb">

您只需要创建一个变量,并通过对测试结果进行分组来构建一个新节点集,并将一个“平均”节点添加到新列表中。

  <xsl:variable name="results">
     <xsl:for-each-group select="httpSample" group-by="@lb">
        <xsl:variable name="count" select="count(current-group())"/>
        <xsl:variable name="totalTime" select="sum(current-group()/@t)"/>
        <test lb="{current-grouping-key()}" num="{$count}" tot="{$totalTime}" avg="{$totalTime div $count}"/>
     </xsl:for-each-group>
  </xsl:variable>

因此,这里的结果变量将包含一个测试元素列表,每个不同的测试一个,新添加的节点具有平均时间。然后,您可以迭代这些结果,甚至对它们求和。

这是完整的 XSLT

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

      <xsl:variable name="results">
         <xsl:for-each-group select="httpSample" group-by="@lb">
            <xsl:variable name="count" select="count(current-group())"/>
            <xsl:variable name="totalTime" select="sum(current-group()/@t)"/>
            <test lb="{current-grouping-key()}" num="{$count}" tot="{$totalTime}" avg="{$totalTime div $count}"/>
         </xsl:for-each-group>
      </xsl:variable>

      <table>
         <tr>
            <th>Test</th>
            <th>Total Test</th>
            <th>Total Time</th>
            <th>Average Time</th>
         </tr>
         <xsl:for-each select="$results/test">
            <tr>
               <td>
                  <xsl:value-of select="@lb"/>
               </td>
               <td>
                  <xsl:value-of select="@num"/>
               </td>
               <td>
                  <xsl:value-of select="@tot"/>
               </td>
               <td>
                  <xsl:value-of select="@avg"/>
               </td>
            </tr>
         </xsl:for-each>
         <tr>
            <td>Total</td>
            <td>
               <xsl:value-of select="sum($results/test/@num)"/>
            </td>
            <td>
               <xsl:value-of select="sum($results/test/@tot)"/>
            </td>
            <td>
               <xsl:value-of select="sum($results/test/@avg)"/>
            </td>
         </tr>
      </table>
   </xsl:template>
</xsl:stylesheet>

当应用于您的 XML 示例时,输出如下

<table>
   <tr>
      <th>Test</th>
      <th>Total Test</th>
      <th>Total Time</th>
      <th>Average Time</th>
   </tr>
   <tr>
      <td>html</td>
      <td>2</td>
      <td>78</td>
      <td>39</td>
   </tr>
   <tr>
      <td>userRoleRetriever</td>
      <td>2</td>
      <td>47</td>
      <td>23.5</td>
   </tr>
   <tr>
      <td>UserActivityWSDL</td>
      <td>2</td>
      <td>47</td>
      <td>23.5</td>
   </tr>
   <tr>
      <td>SubscriberMgmtWSDL</td>
      <td>2</td>
      <td>188</td>
      <td>94</td>
   </tr>
   <tr>
      <td>NetworkMgmtWSDL</td>
      <td>2</td>
      <td>62</td>
      <td>31</td>
   </tr>
   <tr>
      <td>AlarmMgmtWSDL</td>
      <td>2</td>
      <td>31</td>
      <td>15.5</td>
   </tr>
   <tr>
      <td>getSubscribers</td>
      <td>2</td>
      <td>391</td>
      <td>195.5</td>
   </tr>
   <tr>
      <td>getMpegResultsById</td>
      <td>2</td>
      <td>15719</td>
      <td>7859.5</td>
   </tr>
   <tr>
      <td>getOverallSummary</td>
      <td>2</td>
      <td>30</td>
      <td>15</td>
   </tr>
   <tr>
      <td>Total</td>
      <td>18</td>
      <td>16593</td>
      <td>8296.5</td>
   </tr>
</table>
于 2012-06-05T14:29:47.787 回答