59 const GUM_SCALAR& number,
60 const int64_t& den_max,
61 const GUM_SCALAR& zero) {
62 bool isNegative = (number < 0) ?
true :
false;
63 GUM_SCALAR pnumber = (isNegative) ? -number : number;
65 if (std::abs(pnumber - GUM_SCALAR(1.)) < zero) {
66 numerator = (isNegative) ? -1 : 1;
69 }
else if (pnumber < zero) {
75 int64_t a(0), b(1), c(1), d(1);
78 while (b <= den_max && d <= den_max) {
79 mediant = (GUM_SCALAR)(a + c) / (GUM_SCALAR)(b + d);
81 if (std::fabs(pnumber - mediant) < zero) {
82 if (b + d <= den_max) {
83 numerator = (isNegative) ? -(a + c) : (a + c);
87 numerator = (isNegative) ? -c : c;
91 numerator = (isNegative) ? -a : a;
95 }
else if (pnumber > mediant) {
105 numerator = (isNegative) ? -c : c;
109 numerator = (isNegative) ? -a : a;
117 int64_t& denominator,
118 const GUM_SCALAR& number,
119 const double& zero) {
120 const GUM_SCALAR pnumber = (number > 0) ? number : -number;
123 GUM_SCALAR rnumber = pnumber;
126 std::vector< uint64_t > p({0, 1});
127 std::vector< uint64_t > q({1, 0});
130 std::vector< uint64_t > a;
132 uint64_t p_tmp, q_tmp;
135 double delta, delta_tmp;
143 a.push_back(std::lrint(std::floor(rnumber)));
144 p.push_back(a.back() * p.back() + p[p.size() - 2]);
145 q.push_back(a.back() * q.back() + q[q.size() - 2]);
147 delta = std::fabs(pnumber - (GUM_SCALAR)p.back() / q.back());
150 numerator = (int64_t)p.back();
151 if (number < 0) numerator = -numerator;
152 denominator = q.back();
156 if (std::abs(rnumber - a.back()) < 1e-6)
break;
158 rnumber = GUM_SCALAR(1.) / (rnumber - a.back());
161 if (a.size() < 2)
return;
166 Idx i =
Idx(p.size() - 2);
172 p_tmp = n * p[i] + p[i - 1];
173 q_tmp = n * q[i] + q[i - 1];
175 delta = std::fabs(pnumber - ((
double)p[i]) / q[i]);
176 delta_tmp = std::fabs(pnumber - ((
double)p_tmp) / q_tmp);
179 numerator = (int64_t)p[i];
180 if (number < 0) numerator = -numerator;
185 if (delta_tmp < zero) {
186 numerator = (int64_t)p_tmp;
187 if (number < 0) numerator = -numerator;
195 for (n = (a[i - 1] + 2) / 2; n < a[i - 1]; ++n) {
196 p_tmp = n * p[i] + p[i - 1];
197 q_tmp = n * q[i] + q[i - 1];
199 delta_tmp = std::fabs(pnumber - ((
double)p_tmp) / q_tmp);
201 if (delta_tmp < zero) {
202 numerator = (int64_t)p_tmp;
203 if (number < 0) numerator = -numerator;
214 int64_t& denominator,
215 const GUM_SCALAR& number,
216 const int64_t& den_max) {
217 const GUM_SCALAR pnumber = (number > 0) ? number : -number;
219 const uint64_t denMax = (uint64_t)den_max;
222 GUM_SCALAR rnumber = pnumber;
225 std::vector< uint64_t > p({0, 1});
226 std::vector< uint64_t > q({1, 0});
229 std::vector< uint64_t > a;
231 uint64_t p_tmp, q_tmp;
234 double delta, delta_tmp;
238 a.push_back(std::lrint(std::floor(rnumber)));
240 p_tmp = a.back() * p.back() + p[p.size() - 2];
241 q_tmp = a.back() * q.back() + q[q.size() - 2];
243 if (q_tmp > denMax || p_tmp > denMax)
break;
248 if (std::fabs(rnumber - a.back()) < 1e-6)
break;
250 rnumber = GUM_SCALAR(1.) / (rnumber - a.back());
253 if (a.size() < 2 || q.back() == denMax || p.back() == denMax) {
254 numerator = p.back();
255 if (number < 0) numerator = -numerator;
256 denominator = q.back();
263 Idx i =
Idx(p.size() - 1);
268 for (n = a[i - 1] - 1; n >= (a[i - 1] + 2) / 2; --n) {
269 p_tmp = n * p[i] + p[i - 1];
270 q_tmp = n * q[i] + q[i - 1];
272 if (q_tmp > denMax || p_tmp > denMax)
continue;
274 numerator = (int64_t)p_tmp;
275 if (number < 0) numerator = -numerator;
282 p_tmp = n * p[i] + p[i - 1];
283 q_tmp = n * q[i] + q[i - 1];
285 delta_tmp = std::fabs(pnumber - ((
double)p_tmp) / q_tmp);
286 delta = std::fabs(pnumber - ((
double)p[i]) / q[i]);
288 if (delta_tmp < delta && q_tmp <= denMax && p_tmp <= denMax) {
289 numerator = (int64_t)p_tmp;
290 if (number < 0) numerator = -numerator;
293 numerator = (int64_t)p[i];
294 if (number < 0) numerator = -numerator;
static void continuedFracBest(int64_t &numerator, int64_t &denominator, const GUM_SCALAR &number, const int64_t &den_max=1000000)
Find the best rational approximation.
static void continuedFracFirst(int64_t &numerator, int64_t &denominator, const GUM_SCALAR &number, const double &zero=1e-6)
Find the first best rational approximation.
static void farey(int64_t &numerator, int64_t &denominator, const GUM_SCALAR &number, const int64_t &den_max=1000000L, const GUM_SCALAR &zero=1e-6)
Find the rational close enough to a given ( decimal ) number in [-1,1] and whose denominator is not h...