酒日記

2002 09 04

Wed, 04 Sep 2002

秋味 (6% × 1000 ml)

Template-Toolkit 続き。

Template-Toolkit をいじっていて、なんだか動作がもっさりしているような気がしたのでベンチマークを取ってみる。

ベンチマークスクリプトは

#!/usr/bin/perl

use Benchmark;
use HTML::Template;
use Template;
use strict;

my $prefix = $ARGV[0];

my %vars = (
    test_val  => join('', (0..9, "a".."z", "A".."Z")),
    test_loop => [
        { val => 1 },
        { val => qq|酒日記| },
        { val => "http://sake-nikki.dyndns.org/cgi-bin/index.cgi?rm=hogehoge" },
    ],
    test_if   => (rand > 0.5) ? 1 : 0,
);

sub test_tt
{
    my $tt = Template->new({
        INCLUDE_PATH => "./",
    });
    my $output;
    $tt->process($prefix. "test.tt", \%vars, \$output)
        or die $tt->error;
}

sub test_tp
{
    my $tp = HTML::Template->new(
        filename          => $prefix. "test.tp",
        global_vars       => 1,
        die_on_bad_params => 0,
    );
    my $output;
    $tp->param(%vars);
    $output = $tp->output;
}

timethese(100, {
    'HTML::Template  ' => \&test_tp,
    'Template-Toolkit' => \&test_tt,
});

こんな感じ。テンプレートは test.tt (Template-Toolkit用) が

test_val=[% test_val %]
[% FOREACH l = test_loop %]
    val=[% l.val %]
    val_html=[% l.val | html %]
[% END %]
[% IF test_if %]true[% ELSE %]false[% END %]

で、test.tp (HTML::Template) は、

test_val=<TMPL_VAR name="test_val">
<TMPL_LOOP name="test_loop">
    val=<TMPL_VAR name="val">
    val_html=<TMPL_VAR name="val" escape="html">
</TMPL_LOOP>
<TMPL_IF name="test_if">true<TMPL_ELSE>false</TMPL_IF>

これ。動作としては同等品。更に

  • test.tt, test.tp を10個連結した large_test.t[tp]
  • test.tt, test.tp を100個連結した huge_test.t[tp]
  • test.t[tp] とHTML文書(酒日記を適当に抜粋して 1600 byteほどにしたもの) を、3個づつ連結した real_test.t[tp]

を用意。

...

ベンチマーク結果

test.t[tp]
HTML::Template  :  0 wallclock secs ( 0.42 usr +  0.01 sys =  0.43 CPU) @ 232.56/s (n=100)
Template-Toolkit:  2 wallclock secs ( 1.83 usr +  0.02 sys =  1.85 CPU) @ 54.05/s (n=100)

large_test.t[tp]
HTML::Template  :  4 wallclock secs ( 4.74 usr +  0.01 sys =  4.75 CPU) @ 21.05/s (n=100)
Template-Toolkit: 12 wallclock secs (11.85 usr +  0.08 sys = 11.93 CPU) @  8.38/s (n=100)

real_test.t[tp]
HTML::Template  :  1 wallclock secs ( 1.75 usr +  0.03 sys =  1.78 CPU) @ 56.18/s (n=100)
Template-Toolkit:  5 wallclock secs ( 4.40 usr +  0.05 sys =  4.45 CPU) @ 22.47/s (n=100)

huge_test.t[tp]
HTML::Template  : 267 wallclock secs (266.37 usr +  0.08 sys = 266.45 CPU) @  0.38/s (n=100)
Template-Toolkit: 139 wallclock secs (138.85 usr +  0.27 sys = 139.12 CPU) @  0.72/s (n=100)

huge では Templat-Toolkit が HTML::Template を上まわるものの、それ以外では惨敗。 HTML::Template の圧勝。

両者ともに、一旦コンパイルしたテンプレートをキャッシュすることで高速化ができる (特に mod_perl 環境で有効) ので、キャッシュを有効にして再度ベンチマークを取ってみる。 具体的には、

 my $tt = Template->new({
     INCLUDE_PATH => "./",
+    COMPILE_DIR  => "/var/tmp/tt/",
+    COMPILE_EXT  => ".ttc",
 });

 my $tp = HTML::Template->new(
     filename          => $prefix. "test.tp",
     global_vars       => 1,
     die_on_bad_params => 0,
+    cache             => 1,
 );

としてキャッシュを有効にする。結果は以下。

test.t[tp]
HTML::Template  :  0 wallclock secs ( 0.24 usr +  0.00 sys =  0.24 CPU) @ 416.67/s (n=100)
Template-Toolkit:  0 wallclock secs ( 0.71 usr +  0.02 sys =  0.73 CPU) @ 136.99/s (n=100)

large_test.t[tp]
HTML::Template  :  3 wallclock secs ( 3.67 usr +  0.00 sys =  3.67 CPU) @ 27.25/s (n=100)
Template-Toolkit:  3 wallclock secs ( 2.94 usr +  0.03 sys =  2.97 CPU) @ 33.67/s (n=100)

real_test.t[tp]
HTML::Template  :  0 wallclock secs ( 0.65 usr +  0.01 sys =  0.66 CPU) @ 151.52/s (n=100)
Template-Toolkit:  2 wallclock secs ( 1.30 usr +  0.02 sys =  1.32 CPU) @ 75.76/s (n=100)

huge_test.t[tp]
HTML::Template  : 274 wallclock secs (274.28 usr +  0.01 sys = 274.29 CPU) @  0.36/s (n=100)
Template-Toolkit: 35 wallclock secs (33.95 usr +  0.24 sys = 34.19 CPU) @  2.92/s (n=100)

簡単にまとめると、

  • 単純な(小さい)テンプレートでは、HTML::Template が速い
  • Template-Toolkit は、キャッシュによる速度向上が大きい
  • テンプレート上での展開要素が多い場合、Template-Toolkit が有利 (テンプレートが巨大になってもパフォーマンスが落ちにくい)
  • 実運用環境では、ほぼ互角

というところでしょうか。HTML::Template のほうがシンプルな分、速度的には有利な感じ。 Template-Toolkit が明らかに高速だった huge_ は、実際の Web アプリケーションではまずありえない (特殊な) テンプレートだし。


一番絞り (5% × 850 ml)


powered by blosxom