diff options
-rw-r--r-- | lib/Math/Decimal/FastPP.pm | 58 |
1 files changed, 47 insertions, 11 deletions
diff --git a/lib/Math/Decimal/FastPP.pm b/lib/Math/Decimal/FastPP.pm index 538fb92..0f9960d 100644 --- a/lib/Math/Decimal/FastPP.pm +++ b/lib/Math/Decimal/FastPP.pm @@ -4,7 +4,7 @@ use strict; use warnings; use Exporter qw(import); -our @EXPORT_OK = qw(dadd dmul drhtz drhfz); +our @EXPORT_OK = qw(dadd dmul drhtz drhfz dfmt); our $VERSION = '0.001'; @@ -83,6 +83,21 @@ sub drhfz substr($as, length($as) - $_[1]); } +sub dfmt +{ + my ($a, $p) = @_; + my $s; + my $i; + my $f; + my $l; + + ($s, $i, $f) = $a =~ m/^(-?)(\d*).?(\d*)$/; + $l = length($f); + if ($i eq "") { $i = "0"; } + if ($l < $p) { $f .= "0" x ($p - $l); } + return $s . $i . "." . $f; +} + 1; __END__ @@ -97,11 +112,12 @@ Math::Decimal::FastPP - Fast pure-Perl decimal math =head1 SYNOPSIS - use Math::Decimal::FastPP qw(dadd dmul drhtz drhfz); + use Math::Decimal::FastPP qw(dadd dmul drhtz drhfz dfmt); $a = dadd($a, "1.23"); # $a += 1.23 $c = dmul($a, $b); # $c = $a * $b $a = drhtz($a); # Round half toward zero + $a = dfmt($a, 2); # Format with leading/trailing zeroes as needed =head1 DESCRIPTION @@ -111,8 +127,8 @@ built-in binary floating-point math, but they're faster than L<Math::BigFloat|Math::BigFloat> and other commonly used decimal math modules. This module is currently less complete than Perl's built-in math and other -decimal math modules. So far it only includes addition, multiplication, and two -rounding functions. +decimal math modules. So far it only includes addition, multiplication, two +rounding functions, and one formatting function. Despite the similar name and purpose, this module is not compatible with L<Math::Decimal|Math::Decimal>. @@ -133,9 +149,9 @@ zero" and "decimal round (away) from zero". =head1 THEORY The binary operation functions (C<dadd()> and C<dmul()>) operate on two numbers, -C<a> and C<b>. And obviously the unary operation functions (C<drhtz()> and -C<drhfz()>) operate on one number, C<a>. The binary operation functions produce -a resulting number, C<c>. +C<a> and C<b>. And obviously the unary operation functions (C<drhtz()>, +C<drhfz()>, and C<dfmt()>) operate on one number, C<a>. The binary operation +functions produce a resulting number, C<c>. The input numbers C<a> and C<b> are broken into their integer (C<i>) and fractional (C<f>) parts, for example C<$ai> and C<$af>. All three numbers C<a>, @@ -203,6 +219,26 @@ example: C<$p> is a non-negative (i.e. zero or positive) integer representing the number of significant digits right of the radix point. +=head2 dfmt() + + $a = dfmt($a, $p); + +Formats C<$a> with precision C<$p>. Ensures that C<$a> has a leading zero if +its absolute value is less than 1 and ensures that C<$a> has at least C<$p> +digits in its fractional part. For example: + + dfmt("2.3", 2); # Returns "2.30" + dfmt(".23", 2); # Returns "0.23" + +C<$p> is a non-negative (i.e. zero or positive) integer representing the number +of significant digits right of the radix point. + +The following two lines are equivalent, except that C<dfmt()> safely operates on +C<$a> as a string instead of as a floating-point number: + + $a = dfmt($a, $p); + $a = sprintf("%0.${p}f", $a); + =head1 DIAGNOSTICS This module has no diagnostics. @@ -226,10 +262,10 @@ L<Math::Decimal|Math::Decimal>. These arithmetic functions preserve all significant fractional digits, including trailing zeroes. They also don't always add a leading zero before the radix point for numbers with absolute values less than one. So the output numbers can -look "ugly", like ".123000". This is only an issue if the numbers (which are -returned as strings) are concatenated into other strings or printed without -formatting. If this is an issue in your code, use the outputs as numbers (e.g. -C<$c + 0>) or print with formatting (with C<printf>). +look "ugly", like ".123000". This is only an issue when the numbers (which are +returned as strings) are concatenated into other strings or printed. In these +cases, you can round the numbers with C<drhtz()> or C<drhfz()> and format them +with C<dfmt()>. =head1 AUTHOR |