wip
This commit is contained in:
parent
ea85b783c1
commit
5f121f7b95
5 changed files with 28 additions and 8 deletions
BIN
Project0/src/ciao
Executable file
BIN
Project0/src/ciao
Executable file
Binary file not shown.
|
@ -79,24 +79,26 @@ fraction add_fractions(fraction frac1, fraction frac2)
|
|||
|
||||
fraction sum_fraction_array(fraction frac_array[], int n) {
|
||||
fraction sum = { 0, 1 };
|
||||
print_fraction_array(frac_array, n);
|
||||
for (int i = 0; i < n; i++) {
|
||||
fraction frac = frac_array[i];
|
||||
sum = add_fractions(sum, frac);
|
||||
print_fraction(sum);
|
||||
}
|
||||
return sum;
|
||||
}
|
||||
|
||||
double sum_fraction_array_approx(fraction frac_array[], int n)
|
||||
{
|
||||
return fraction2double(sum_fraction_array(frac_array, n));
|
||||
double sum = 0;
|
||||
for (int i = 0; i < n; i++) {
|
||||
sum += fraction2double(frac);
|
||||
}
|
||||
return sum;
|
||||
|
||||
// the approximation in this function is given by the fact that floating
|
||||
// point numbers cannot represent precisely rational numbers due to the
|
||||
// fixed size of significant digits they can hold in the mantissa. This
|
||||
// implementation minimizes the approximation by computing the sum into a
|
||||
// fraction FIRST before performing the double conversion.
|
||||
// fixed size of significant digits they can hold in the mantissa. The
|
||||
// approximation adds up for each fraction to double conversion performed in
|
||||
// the loop
|
||||
}
|
||||
|
||||
void fill_fraction_array(fraction frac_array[], int n)
|
||||
|
|
BIN
Project0/src/fraction_summing/fraction_toolbox.o
Normal file
BIN
Project0/src/fraction_summing/fraction_toolbox.o
Normal file
Binary file not shown.
BIN
Project0/src/fraction_summing/main
Executable file
BIN
Project0/src/fraction_summing/main
Executable file
Binary file not shown.
|
@ -1,6 +1,7 @@
|
|||
#include <iostream>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <limits.h>
|
||||
|
||||
#include "fraction_toolbox.hpp"
|
||||
|
||||
|
@ -71,7 +72,18 @@ static void test_array_functions(int n)
|
|||
cout << "Sum of array as double (approx.): " << endl;
|
||||
cout << sum_fraction_array_approx(a, n) << endl;
|
||||
|
||||
//TODO: find n for which sum function breaks. Explain what is happening.
|
||||
// find n for which sum function breaks. Explain what is happening. n=4448,
|
||||
// this is the first number where the numerator overflows and thus
|
||||
// erroneoulsly becomes a negative number. The overflow is caused by the
|
||||
// sum_fractions function, specifically in the multiplication between
|
||||
// denominators. The approximation function doesn't overflow since fractions
|
||||
// are first converted into double floating point numbers, thus never
|
||||
// calling the sum_fractions functions and performing dangerous fixed size
|
||||
// integer multiplications. Additionally, due to the floating point nature
|
||||
// of the double type, a sum of doubles will never overflow to something
|
||||
// "weird", albeit precision will be lost due to the limited number of
|
||||
// significant digits and for larger and larger values the sum will
|
||||
// eventually reach the value "+Infinity" due to the fixed exponent size.
|
||||
}
|
||||
|
||||
static void test_toolbox(int argc, char* argv[])
|
||||
|
@ -83,7 +95,13 @@ static void test_toolbox(int argc, char* argv[])
|
|||
test5();
|
||||
|
||||
cout << "\n========== test_array_functions ========= " << endl;
|
||||
#if 1
|
||||
cout << "Input n: " << endl;
|
||||
int n;
|
||||
cin >> n;
|
||||
#else
|
||||
int n = 5;
|
||||
#endif
|
||||
test_array_functions(n);
|
||||
}
|
||||
|
||||
|
|
Reference in a new issue