Assortment of Ada functions to solve Project Euler problems.
Euler Tools is an Ada 2022 library of functions that contains usual mathematical functions and non-usual numeric functions, like concatenation of numbers, palindrome checking or the length (number of digits) of a number.
New functions are added to the library from each solved problem: common functionality that can be abstracted, and possibly reused in other problems, becomes an Euler Tools function.
For example, to solve problem 3 you
need to deal with prime numbers and to check if a given number evenly divides
another one. Also, to reduce the search space, divisors checked start with
the squared root of a number. Thus, three functions are used from Euler
Tools: Square_Root
, Is_Divisor
and Is_Prime
.
Euler Tools is currently in its very early stage.
New functions are being added very often, some are re-implemented, some optimized, or specifications change according to some criteria. But to make things clear, the set of unit tests included in the library can be used to understand a function specification.
New Ada 2022 features are expected to be supported by compilers soon. Some features are really interesting to Euler Tools, like parallel blocks, and it will be used as they are available.
Ada pre- and post-conditions are used in some function specifications. The plan is to improve and generalize its use. SPARK is also planned to be used.
Use Alire to get and compile the library:
alr get euler_tools
cd euler_tools*
alr build
To compile and run the unit tests:
cd tests
alr build
./bin/euler_tools_tests
Traceability matrix of functions vs unit tests is provided here.
Main component of Euler Tools is a generic package, Euler_Package (<>)
,
that can be instantiated with any integer type, from Integer
to
Long_Long_Long_Integer
.
Some instantiations are provided as separate packages:
package Euler_Tools is new Euler_Package (Integer)
package Euler_Tools_Int1 is new Euler_Package (Long_Integer)
package Euler_Tools_Int2 is new Euler_Package (Long_Long_Integer)
package Euler_Tools_Int3 is new Euler_Package (Long_Long_Long_Integer)
Each platform provides different representation of type, ranging from 32 or
64 bits for Integer
, or 128 bits for Long_Long_Long_Integer
.
Each package provides also the Integer_Type
that has been used to
instantiate Euler_Package (<>)
. This is a convenient type to easily
change from one package to another without having to rewrite the integer
type used. For example, you can start solving a problem with
Euler_Tools
:
with Text_IO; use Text_IO;
with Euler_Tools; use Euler_Tools;
procedure Problem_42 is
Answer : Integer_Type := Factorial (12);
begin
Put_Line ("Answer:" & Answer'Image);
end Problem_42;
If later on you realize that Integer
is not big enough, the you can try
with a longer type simply by replacing the Euler package:
with Text_IO; use Text_IO;
with Euler_Tools_Int3; use Euler_Tools_Int3;
procedure Problem_42 is
Answer : Integer_Type := Factorial (33);
begin
Put_Line ("Answer:" & Answer'Image);
end Problem_42;
Other packages included in Euler Tools are:
Euler_Calendar
, that provides a function to get the weekday of a date (YYYY/MM/DD).Euler_Text
, that implements a function that return a number written out in words.
Solving mathematical problems usually require not only Integer_Type
, but
more complex types. Euler Tools contains the following types for your
convenience:
-
List_Type
is a list ofInteger_Type
. It is a doubly linked list, so elements can be visited in both directions (seeAda.Containers.Doubly_Linked_Lists
). -
Set_Type
is a set ofInteger_Type
. You canInsert
new numbers,Include
possibly repeated numbers, make it anEmpty_Set
, get theLength
(cardinality) of a set, etc. It is an ordered set, so elements of a set are always visited in order (seeAda.Containers.Ordered_Sets
). -
Numeral_Type
isrange 0 .. 9
(4 bits long), a subtype used in other types. -
Crumbled_Natural
is an arbitrary long natural number composed as a vector of numerals. Contrast with Ada 2022 packageAda.Numerics.Big_Numbers.Big_Integers
. -
Decimal_Division_Type
is used to perform arbitrary precision divisions. For example, problem 26 requires really long divisions (spoiler: recurring cycle of$1/n$ contains 982 digits, for some$n<1000$ ).
Please check some Project Euler solutions provided as example of use of Euler Tools in Euler Examples repository.
Please feel free to
- Use Euler Tools to solve Project Euler or other mathematical problems: send us the link to you work and we'll it include in this section.
- Send contributions, ideas and suggestions to improve Euler Tools functions or implementations.
MIT (c) 2023 Francesc Rocher