Numeric double cast/comparison

This commit is contained in:
Klaus Basan
2018-08-16 02:49:33 +02:00
parent 3a09749103
commit 660a1adb71
4 changed files with 31 additions and 24 deletions

View File

@@ -22,21 +22,19 @@ namespace BlackMisc
{
namespace Math
{
double CMathUtils::hypot(double x, double y)
{
x = qAbs(x);
y = qAbs(y);
double max = std::max(x, y);
double min = std::min(x, y);
const double max = std::max(x, y);
const double min = std::min(x, y);
double r = min / max;
return max * sqrt(1 + r * r);
}
double CMathUtils::cubicRootReal(double x)
{
double result;
result = std::pow(qAbs(x), 1.0 / 3.0);
const double result = std::pow(qAbs(x), 1.0 / 3.0);
return x < 0 ? -result : result;
}
@@ -45,10 +43,10 @@ namespace BlackMisc
// gosh, is there no Qt method for this??? It's year 2013
double fractpart, intpart;
fractpart = modf(value, &intpart);
if (fractpart == 0) return value; // do not mess any "integers" to the worse
double m = pow(10.0, digits);
qint64 ri = qRound64(value * m); // do not loose any range here
double rv = static_cast<double>(ri) / m;
if (epsilonZeroLimits(fractpart)) { return value; } // do not mess any "integers" to the worse
const double m = pow(10.0, digits);
const qint64 ri = qRound64(value * m); // do not loose any range here
const double rv = static_cast<double>(ri) / m;
return rv;
}
@@ -59,19 +57,18 @@ namespace BlackMisc
double CMathUtils::roundEpsilon(double value, double epsilon)
{
if (epsilon == 0) { return value; } // avoid division by 0
if (epsilonZeroLimits(epsilon)) { return value; } // avoid division by 0
double fractpart, intpart;
fractpart = modf(value, &intpart);
if (fractpart == 0) { return value; } // do not mess any "integers" to the worse
if (epsilonZeroLimits(fractpart)) { return value; } // do not mess any "integers" to the worse
const double roundValue = value / epsilon;
qint64 ri = qRound64(roundValue);
double rv = static_cast<double>(ri) * epsilon; // do not loose any range here
const qint64 ri = qRound64(roundValue);
const double rv = static_cast<double>(ri) * epsilon; // do not loose any range here
return rv;
}
bool CMathUtils::epsilonEqual(double v1, double v2, double epsilon)
{
if (v1 == v2) return true;
return qAbs(v1 - v2) <= epsilon;
}

View File

@@ -17,6 +17,7 @@
#include <QtCore/qmath.h>
#include <QPair>
#include <cmath>
#include <numeric>
namespace BlackMisc
{
@@ -56,8 +57,16 @@ namespace BlackMisc
//! Round by given epsilon
static double roundEpsilon(double value, double epsilon);
//! Epsilon safe equal
//! Epsilon safe equal @{
static bool epsilonEqual(double v1, double v2, double epsilon = 1E-06);
static bool epsilonEqualLimits(double v1, double v2) { return qAbs(v1 - v2) <= std::numeric_limits<double>::epsilon(); }
//! @}
//! Epsilon safe zero @{
static bool epsilonZero(double v, double epsilon) { return epsilonEqual(v, 0.0, epsilon); }
static bool epsilonZero(double v) { return epsilonEqual(v, 0.0, 1E-09); }
static inline bool epsilonZeroLimits(double v) { return qAbs(v) <= std::numeric_limits<double>::epsilon(); }
//! @}
//! Nearest integer not greater in magnitude than value, correcting for epsilon
static inline double trunc(double value, double epsilon = 1e-10)