slope 0.29.0
Loading...
Searching...
No Matches
qnorm.cpp
1#include <array>
2#include <cmath>
3#include <limits>
4#include <stdexcept>
5
6namespace slope {
7
8double
9normalQuantile(const double p)
10{
11 if (p < 0 || p > 1) {
12 throw std::out_of_range("p must be in the range [0, 1]");
13 }
14
15 if (p == 0) {
16 return -std::numeric_limits<double>::infinity();
17 } else if (p == 1) {
18 return std::numeric_limits<double>::infinity();
19 }
20
21 // Coefficients in rational approximations
22 const std::array<double, 6> a = {
23 -3.969683028665376e+01, 2.209460984245205e+02, -2.759285104469687e+02,
24 1.383577518672690e+02, -3.066479806614716e+01, 2.506628277459239e+00
25 };
26 const std::array<double, 6> b = { -5.447609879822406e+01,
27 1.615858368580409e+02,
28 -1.556989798598866e+02,
29 6.680131188771972e+01,
30 -1.328068155288572e+01 };
31
32 const std::array<double, 6> c = {
33 -7.784894002430293e-03, -3.223964580411365e-01, -2.400758277161838e+00,
34 -2.549732539343734e+00, 4.374664141464968e+00, 2.938163982698783e+00
35 };
36 const std::array<double, 6> d = { 7.784695709041462e-03,
37 3.224671290700398e-01,
38 2.445134137142996e+00,
39 3.754408661907416e+00 };
40
41 // Define break-points
42 const double plow = 0.02425;
43 const double phigh = 1 - plow;
44
45 // Rational approximations for lower region
46 if (p < plow) {
47 double q = std::sqrt(-2 * std::log(p));
48
49 return (((((c[0] * q + c[1]) * q + c[2]) * q + c[3]) * q + c[4]) * q +
50 c[5]) /
51 ((((d[0] * q + d[1]) * q + d[2]) * q + d[3]) * q + 1);
52 }
53 // Rational approximations for upper region
54 if (phigh < p) {
55 double q = std::sqrt(-2 * std::log(1 - p));
56
57 return -(((((c[0] * q + c[1]) * q + c[2]) * q + c[3]) * q + c[4]) * q +
58 c[5]) /
59 ((((d[0] * q + d[1]) * q + d[2]) * q + d[3]) * q + 1);
60 }
61
62 // Rational approximations for central region
63 const double q = p - 0.5;
64 const double r = q * q;
65
66 return (((((a[0] * r + a[1]) * r + a[2]) * r + a[3]) * r + a[4]) * r + a[5]) *
67 q / (((((b[0] * r + b[1]) * r + b[2]) * r + b[3]) * r + b[4]) * r + 1);
68}
69
70} // namespace slope
Namespace containing SLOPE regression implementation.
Definition clusters.cpp:5
double normalQuantile(const double p)
Computes the quantile of a standard normal distribution using the Beasley-Springer-Moro algorithm.
Definition qnorm.cpp:9