—# Copyright 2003 Fergal Daly <fergal@esatclear.ie> distributed under the GNU
# Lesser General Public License, you do not have to accept this license but
# nothing else gives you the right to use this software
use
strict;
use
warnings;
package
Test::Benchmark;
use
Test::Builder;
$VERSION @EXPORT @ISA @CARP_NOT $VERBOSE
)
;
$VERSION
=
"0.004"
;
$VERBOSE
= 0;
my
$Test
= Test::Builder->new;
require
Exporter;
@ISA
=
qw( Exporter )
;
@EXPORT
=
qw( is_faster is_n_times_faster is_fastest )
;
@CARP_NOT
=
qw( Test::Benchmark Benchmark )
;
sub
is_fastest
{
my
$which
=
shift
;
my
$times
=
shift
;
my
$marks
=
shift
;
my
$name
=
shift
;
my
@marks
;
while
(
my
(
$name
,
$sub
) =
each
%$marks
)
{
my
$res
= get_res(
$times
,
$sub
);
my
(
$r
,
$pu
,
$ps
,
$cu
,
$cs
,
$n
) =
@$res
;
push
(
@marks
,
{
name
=>
$name
,
res
=>
$res
,
n
=>
$n
,
s
=> (
$pu
+
$ps
),
}
);
}
@marks
=
sort
{
$b
->{n} *
$a
->{s} <=>
$a
->{n} *
$b
->{s}}
@marks
;
my
$ok
=
$marks
[0]->{name} eq
$which
;
$Test
->ok(
$ok
,
$name
);
if
(not
$ok
)
{
$Test
->diag(
"$which was not the fastest"
);
}
if
(not
$ok
or
$VERBOSE
)
{
foreach
my
$mark
(
@marks
)
{
$Test
->diag(
"$mark->{name} - "
.timestr(
$mark
->{res}));
}
}
}
sub
is_faster
{
local
$Test::Builder::Level
=
$Test::Builder::Level
+ 1;
if
(
ref
$_
[1])
{
is_n_times_faster(1,
@_
);
}
else
{
is_n_times_faster(
@_
);
}
}
sub
is_n_times_faster
{
my
$factor
=
shift
;
my
$times
=
shift
;
my
$code1
=
shift
;
my
$code2
=
shift
;
my
$name
=
shift
;
my
@res
;
my
(
$res1
,
$res2
) =
map
{get_res(
$times
,
$_
)} (
$code1
,
$code2
);
my
(
$r1
,
$pu1
,
$ps1
,
$cu1
,
$cs1
,
$n1
) =
@$res1
;
my
(
$r2
,
$pu2
,
$ps2
,
$cu2
,
$cs2
,
$n2
) =
@$res2
;
# we want code1 to be faster than code2. Essentially we are comparing 2
# fractions factor * n1/cpu1 > n2/cpu2 but in order to avoid div by zero
# we use multiplication
my
$ok
= (
$n1
* (
$pu2
+
$ps2
) *
$factor
) > (
$n2
* (
$pu1
+
$ps1
));
$Test
->ok(
$ok
,
$name
);
if
(not
$ok
)
{
my
$extra
= (
$factor
== 1) ?
""
:
" at least $factor times"
;
$Test
->diag(
"code1 was not$extra faster than code 2"
);
}
if
(not
$ok
or
$VERBOSE
)
{
$Test
->diag(timestr(
$res1
));
$Test
->diag(timestr(
$res2
));
}
return
$ok
;
}
sub
get_res
{
my
(
$times
,
$sub
) =
@_
;
if
(
ref
(
$sub
) eq
"Benchmark"
)
{
return
$sub
;
}
elsif
(0)
{
# get from cache not implemented - maybe never will be...
}
else
{
return
timethis(
$times
,
$sub
,
""
,
"none"
);
}
}
sub
builder
{
if
(
@_
)
{
$Test
=
shift
;
}
return
$Test
;
}
1;
__END__
=head1 NAME
Test::Benchmark - Make sure something really is faster
=head1 SYNOPSIS
use Test::More test => 17;
use Test::Benchmark;
is_faster(-10, sub {...}, sub {...}, "this is faster than that")
is_faster(5, -10, sub {...}, sub {...}, "this is 5 times faster than that")
is_n_times_faster(5, -10, sub {...}, sub {...}, "this is 5 times faster than that")
is_faster(-10, $bench1, $bench2, "res1 was faster than res2");
is_fastest("c", -1, {a => sub {...}, b => sub {...}, c => sub {...},);
=head1 DESCRIPTION
Sometimes you want to make sure that your "faster" algorithm really is
faster than the old way. This lets you check. It might also be useful to
check that your super whizzo XS or Inline::C version is actually faster.
This module is based on the standard L<Benchmark> module. If you have lots
of timings to compare and you don't want to keep running the same benchmarks
all the time, you can pass in a result object from C<Benchmark::timethis()>
instead of sub routine reference.
=head1 USAGE
There are 3 functions exported: C<is_faster()>, C<is_n_times_faster()> and
C<is_fastest()>. Actually C<is_faster()> is redundant because
C<is_n_times_faster()> can do the same thing just by setting n to 1.
Anywhere you can pass a subroutine reference you can also pass in a
L<Benchmark> object.
# call as
# is_faster($times, $sub1, $sub2, $name)
# is_faster($faster, $times, $sub1, $sub2, $name)
If you leave the number of iterations blank then it will use L<Benchmark>'s
default.
=head2 is_faster()
is_faster($times, $sub1, $sub2, $name)
is_faster($factor, $times, $sub1, $sub2, $name)
This runs each subroutine reference C<$times> times and then compares the
results. Instead of either subroutine reference you can pass in a
L<Benchmark> object. If you pass in 2 L<Benchmark> objects then C<$times> is
irrelevant.
If C<$times> is negative then that specifies a minimum duration for the
benchmark rather than a number of iterations (see L<Benchmark> for more
details). B<I strongly recommend you use this feature if you want your
modules to still pass tests reliably on machines that are much faster than
your own.> 10000 iterations may be enough for a reliable benchmark on your
home PC but it be just a twinkling in the eye of somebody else's super
computer.
If the test fails, you will get a diagnostic output showing the benchmark
results in the standard L<Benchmark> format.
=head2 is_n_times_faster()
is_n_times_faster($factor, $times, $sub1, $sub2, $name)
This is exactly the same as the second form of is_faster but it's just
explicit about the "n times" part.
=head2 is_fastest()
is_fastest($bname, $times, $bhash, $name)
This takes a hash reference containing benchmark subroutines as values and
their names as keys. It times each subroutine and checks that the one named
in C<$bname> was the fastest. If the test fails, it will print out a sorted
list of the timings of all the subroutines.
=head1 VARIABLES
If you set C<$Test::Benchmark::VERBOSE = 1> then the benchmark results will
be printed out as diagnostics whether the tests pass or fail.
=head1 DANGERS
Benchmarking can be slow so please consider leaving out the benchmark tests
from your default test suite, perhaps only running them if the user has set
a particular environment variable.
Some benchmarks are inherently unreliable and should not be part of an
automated test suite.
=head1 BUGS
None that I know of but I said that last time too.
=head1 DEPENDENCIES
L<Benchmark>, L<Test::Builder> but they come with most Perls.
=head1 HISTORY
This came up on the perl-qa mailing list, when Jim Cromie wanted to display
benchmark results in tests.
=head1 SEE ALSO
L<Test::Builder>, L<Benchmark>
=head1 AUTHOR
Written by Fergal Daly <fergal@esatclear.ie>.
=head1 COPYRIGHT
Copyright 2003 by Fergal Daly E<lt>fergal@esatclear.ieE<gt>.
This program is free software and comes with no warranty. It is distributed
under the LGPL license. You do not have to accept this license but nothing
else gives you the right to use this software.
See the file F<LGPL> included in this distribution or
=cut