=encoding utf8
=head1 NAME
Math::Bacovia - Symbolic math library with support for alternative representations.
=head1 VERSION
Version 0.04
=head1 SYNOPSIS
B<Math::Bacovia> is an experimental symbolic math library, with support for alternative representations, simplification of expressions and numerical evaluation.
use 5.014;
use Math::Bacovia qw(:all);
my $x = Symbol('x');
my $y = Symbol('y');
say $x+$y; #=> Sum(Symbol("x"), Symbol("y"))
say $x-$y; #=> Difference(Symbol("x"), Symbol("y"))
say $x*$y; #=> Product(Symbol("x"), Symbol("y"))
say $x/$y; #=> Fraction(Symbol("x"), Symbol("y"))
say $x**$y; #=> Power(Symbol("x"), Symbol("y"))
say Log($x); #=> Log(Symbol("x"))
say Log($x)+Log($y); #=> Log(Product(Symbol("x"), Symbol("y")))
say Exp($x); #=> Exp(Symbol("x"))
say Exp($x)*Exp($y); #=> Exp(Sum(Symbol("x"), Symbol("y")))
say "\n=> Sum:";
my $sum = Fraction(0, 1);
for my $n (1..10) {
$sum += Fraction(1, $n);
}
say $sum; #=> Fraction(10628640, 3628800)
say $sum->numeric; #=> 7381/2520
say "\n=> Product:";
my $prod = Product();
for my $n (1..3) {
$prod *= Exp(Fraction(1, $n));
}
say $prod->pretty; #=> (exp(1) * exp(1/2) * exp(1/3))
say $prod->simple->pretty; #=> exp(11/6)
say $prod->numeric; #=> 6.25470095193632871640207...
say "\n=> Alternative representations:";
say join ', ', Power(3, 5)->alternatives(full => 1); #=> Power(3, 5), Exp(Product(Log(3), 5)), 243
=head1 TYPES
The types supported by this library are described bellow:
=head2 Symbol(name, value=undef)
Represents a symbolic value. Optionally, it can have a numerical value (or any other value).
=head2 Number(value)
Represents a numerical value.
=head2 Fraction(numerator, denominator)
Represents a symbolic fraction.
=head2 Difference(minuend, subtrahend)
Represents a symbolic difference.
=head2 Power(base, power)
Represents a symbolic exponentiation in a symbolic base.
=head2 Log(x)
Represents the natural logarithm of a symbolic value.
=head2 Exp(x)
Represents the natural exponentiation of a symbolic value.
=head2 Sum(a, b, c, ...)
Represents a summation of an arbitrary (finite) number of symbolic values.
=head2 Product(a, b, c, ...)
Represents a product of an arbitrary (finite) number of symbolic values.
=head1 EXPORT
Importing a name from B<Math::Bacovia> can be done using the following syntax:
use Math::Bacovia qw(Fraction); # imports the Fraction() function
=head2 i / e / pi / tau
B<Math::Bacovia> also provides a small list of useful constants which can be imported:
i the imaginary unit sqrt(-1)
e the e mathematical constant Exp(1)
pi the PI mathematical constant Log(-1) * -sqrt(-1)
tau the TAU mathmematical constant Log(-1) * -sqrt(-1) * 2
For importing a constant, the same syntax can be used as in importing a function:
use Math::Bacovia qw(i tau);
By specifying the B<:all> special keyword, B<Math::Bacovia> will export all the provided functions and constants.
use Math::Bacovia qw(:all);
Nothing is exported by default.
=head1 METHODS
This section describes the special methods provided by Math::Bacovia.
=head2 inv
1 / $x
$x->inv
Multiplicative inverse of C<x>, defined as:
Fraction(1, $x)
=head2 neg
-$x
$x->neg
Additive inverse of C<x>, defined as:
Difference(0, $x)
=head2 add
$x + $y
$x->add($y)
Summation of C<x> and C<y>, defined as:
Sum($x, $y)
=head2 sub
$x - $y
$x->sub($y)
Difference of C<x> from C<y>, defined as:
Difference($x, $y)
=head2 mul
$x * $y
$x->mul($y)
Product of C<x> and C<y>, defined as:
Product($x, $y)
=head2 div
$x / $y
$x->div($y)
Fraction C<x> over C<y>, defined as:
Fraction($x, $y)
=head2 pow
$x ** $y
$x->pow($y)
C<x> to the power C<y>, defined as:
Power($x, $y)
=head2 sqrt
sqrt($x)
$x->sqrt
Square root of C<x>, defined as:
$x ** Fraction(1, 2)
=head2 log
log($x)
$x->log
Natural logarithm of C<x>, defined as:
Log($x)
=head2 exp
exp($x)
$x->exp
Natural exponentiation of C<x>, defined as:
Exp($x)
=head2 int
int($x)
$x->int
Evaluates C<x> numerically and truncates the result to an integer, returing a L<Math::AnyNum> object.
It may return C<NaN> when this conversion is not possible.
Defined as:
$x->numeric->int
=head1 TRIGONOMETRIC METHODS
=head2 sin / sinh / asin / asinh
sin($x)
$x->sin
$x->sinh
$x->asin
$x->asinh
Sine, hyperbolic sine, inverse sine and inverse hyperbolic sine.
=head2 cos / cosh / acos / acosh
cos($x)
$x->cos
$x->cosh
$x->acos
$x->acosh
Cosine, hyperbolic cosine, inverse cosine and inverse hyperbolic cosine.
=head2 tan / tanh / atan / atanh
$x->tan
$x->tanh
$x->atan
$x->atanh
Tangent, hyperbolic tangent, inverse tangent and inverse hyperbolic tangent.
=head2 cot / coth / acot / acoth
$x->cot
$x->coth
$x->acot
$x->acoth
Cotangent, hyperbolic cotangent, inverse cotangent and inverse hyperbolic cotangent.
=head2 sec / sech / asec / asech
$x->sec
$x->sech
$x->asec
$x->asech
Secant, hyperbolic secant, inverse secant and inverse hyperbolic secant.
=head2 csc / csch / acsc / acsch
$x->csc
$x->csch
$x->acsc
$x->acsch
Cosecant, hyperbolic cosecant, inverse cosecant and inverse hyperbolic cosecant.
=head2 atan2
atan2($x, $y)
$x->atan2($y)
The arc tangent of C<x> and C<y>, defined as:
atan2(x, y) = -i * log((y + x*i) / sqrt(x^2 + y^2))
=head1 SPECIAL METHODS
Each type provided by B<Math::Bacovia> includes the same set of methods, defined bellow:
=head2 alternatives
$x->alternatives(%opt)
This method uses common mathematical identities to create symbolically equivalent expressions from the self-expression.
Example:
say for Exp(Log(Fraction(1,3)) * 2)->alternatives;
Output:
Exp(Product(2, Log(Fraction(1, 3))))
Power(Fraction(1, 3), 2)
Exp(Product(2, Log(1/3)))
Power(1/3, 2)
The options supported by this method are:
log => 1, # will try to generate logarithmic alternatives
full => 1, # will try to generate more alternatives (it may be slow)
Example:
say for Power(3, 5)->alternatives(full => 1);
Output:
Power(3, 5)
Exp(Product(Log(3), 5))
243
B<WARNING:> The number of alternative representations grows exponentially! For non-trivial expressions,
this process may take a very long time and use lots of memory. In combination with the B<full> option
(set to a true value), the returned list may contain hundreds or even thousands of alternative representations.
=head2 simple
$x->simple(%opt)
Returns a simplification of the self-expression, taking the same options as the C<alternatives()> method.
say Exp(Log(Log(Exp(Exp(Log(Symbol('x')))))))->simple;
Output:
Symbol("x")
The simplification of an expression is done by generating the list of
alternative representations for it, then selecting the shortest expression from this list.
This approach works pretty well even on relatively complex expressions, such as:
say Log(Symbol('x'))->cosh->simple->pretty; #=> ((1 + x^2)/(2 * x))
However, the returned simplified expression is not guaranteed to always be "optimal". In some cases, calling the C<simple()>
method on an already simplified expression, the simplified expression may get simplified even further:
$x->simple(full => 1)->simple(full => 1);
B<WARNING:> On long expressions, this process may take a very long time and use lots of memory.
=head2 expand
$x->expand(%opt)
Returns an expanded version of the self-expression, taking the same options as the C<alternatives()> method.
say Power(Fraction(5, 7), Fraction(1, 3))->expand(full => 1);
Output:
Exp(Product(Log(Fraction(5, 7)), Fraction(1, 3)))
The expansion of an expression is done by generating the list of
alternative representations for it, then selecting the longest expression from this list.
B<WARNING:> On long expressions, this process may take a very long time and use lots of memory.
=head2 numeric
$x->numeric
Evaluates the self-expression numerically and returns the result as a L<Math::AnyNum> object.
Example:
my $x = Symbol('x', 13);
my $expr = ($x**2 - $x + 41);
say $expr->numeric; #=> 197
=head2 pretty
$x->pretty
Returns a human-readable stringification of the self-expression.
Example:
say Power(3, Log(Fraction(1, 2)))->pretty; #=> 3^log(1/2)
=head1 SEE ALSO
For a detailed documentation of each class, please see:
=over 4
=item * L<Math::Bacovia::Fraction>
=item * L<Math::Bacovia::Difference>
=item * L<Math::Bacovia::Power>
=item * L<Math::Bacovia::Log>
=item * L<Math::Bacovia::Exp>
=item * L<Math::Bacovia::Sum>
=item * L<Math::Bacovia::Product>
=item * L<Math::Bacovia::Number>
=item * L<Math::Bacovia::Symbol>
=back
=head1 AUTHOR
Daniel Șuteu, C<< <trizen at protonmail.com> >>
=head1 BUGS
Please report any bugs or feature requests to L<https://github.com/trizen/Math-Bacovia>.
I will be notified, and then you'll automatically be notified of progress on your bug as I make changes.
=head1 SUPPORT
You can find documentation for this module with the perldoc command.
perldoc Math::Bacovia
You can also look for information at:
=over 4
=item * AnnoCPAN: Annotated CPAN documentation
=item * CPAN Ratings
=item * Search CPAN
=item * GitHub
=back
=head1 LICENSE AND COPYRIGHT
Copyright 2017-2018 Daniel Șuteu.
This program is free software; you can redistribute it and/or modify it
under the terms of the the Artistic License (2.0). You may obtain a
copy of the full license at:
Any use, modification, and distribution of the Standard or Modified
Versions is governed by this Artistic License. By using, modifying or
distributing the Package, you accept this license. Do not use, modify,
or distribute the Package, if you do not accept this license.
If your Modified Version has been derived from a Modified Version made
by someone other than you, you are nevertheless required to ensure that
your Modified Version complies with the requirements of this license.
This license does not grant you the right to use any trademark, service
mark, tradename, or logo of the Copyright Holder.
This license includes the non-exclusive, worldwide, free-of-charge
patent license to make, have made, use, offer to sell, sell, import and
otherwise transfer the Package with respect to any patent claims
licensable by the Copyright Holder that are necessarily infringed by the
Package. If you institute patent litigation (including a cross-claim or
counterclaim) against any party alleging that the Package constitutes
direct or contributory patent infringement, then this Artistic License
to you shall terminate on the date that such litigation is filed.
Disclaimer of Warranty: THE PACKAGE IS PROVIDED BY THE COPYRIGHT HOLDER
AND CONTRIBUTORS "AS IS' AND WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES.
THE IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
PURPOSE, OR NON-INFRINGEMENT ARE DISCLAIMED TO THE EXTENT PERMITTED BY
YOUR LOCAL LAW. UNLESS REQUIRED BY LAW, NO COPYRIGHT HOLDER OR
CONTRIBUTOR WILL BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, OR
CONSEQUENTIAL DAMAGES ARISING IN ANY WAY OUT OF THE USE OF THE PACKAGE,
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
=cut