All Downloads are FREE. Search and download functionalities are using the official Maven repository.

smile.math.special.Beta Maven / Gradle / Ivy

The newest version!
/*******************************************************************************
 * Copyright (c) 2010 Haifeng Li
 *   
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *  
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 *******************************************************************************/
package smile.math.special;

import smile.math.Math;

/**
 * The beta function, also called the Euler integral of the first kind.
 * 

* B(x, y) = 01 tx-1 (1-t)y-1dt *

* for x, y > 0 and the integration is over [0,1].The beta function is symmetric, i.e. B(x,y) = B(y,x). * * @author Haifeng Li */ public class Beta { /** * A small number close to the smallest representable floating point number. */ private static final double FPMIN = 1e-300; /** * Beta function, also called the Euler integral of the first kind. * The beta function is symmetric, i.e. B(x,y)==B(y,x). */ public static double beta(double x, double y) { return Math.exp(Gamma.logGamma(x) + Gamma.logGamma(y) - Gamma.logGamma(x + y)); } /** * Regularized Incomplete Beta function. * Continued Fraction approximation (see Numerical recipies for details) */ public static double regularizedIncompleteBetaFunction(double alpha, double beta, double x) { if (x < 0.0 || x > 1.0) { throw new IllegalArgumentException("Invalid x: " + x); } double ibeta = 0.0; if (x == 0.0) { ibeta = 0.0; } else { if (x == 1.0) { ibeta = 1.0; } else { // Term before continued fraction ibeta = Math.exp(Gamma.logGamma(alpha + beta) - Gamma.logGamma(alpha) - Gamma.logGamma(beta) + alpha * Math.log(x) + beta * Math.log(1.0D - x)); // Continued fraction if (x < (alpha + 1.0) / (alpha + beta + 2.0)) { ibeta = ibeta * incompleteFractionSummation(alpha, beta, x) / alpha; } else { // Use symmetry relationship ibeta = 1.0 - ibeta * incompleteFractionSummation(beta, alpha, 1.0 - x) / beta; } } } return ibeta; } /** * Incomplete fraction summation used in the method regularizedIncompleteBeta * using a modified Lentz's method. */ private static double incompleteFractionSummation(double alpha, double beta, double x) { final int MAXITER = 500; final double EPS = 3.0E-7; double aplusb = alpha + beta; double aplus1 = alpha + 1.0; double aminus1 = alpha - 1.0; double c = 1.0; double d = 1.0 - aplusb * x / aplus1; if (Math.abs(d) < FPMIN) { d = FPMIN; } d = 1.0 / d; double h = d; double aa = 0.0; double del = 0.0; int i = 1, i2 = 0; boolean test = true; while (test) { i2 = 2 * i; aa = i * (beta - i) * x / ((aminus1 + i2) * (alpha + i2)); d = 1.0 + aa * d; if (Math.abs(d) < FPMIN) { d = FPMIN; } c = 1.0 + aa / c; if (Math.abs(c) < FPMIN) { c = FPMIN; } d = 1.0 / d; h *= d * c; aa = -(alpha + i) * (aplusb + i) * x / ((alpha + i2) * (aplus1 + i2)); d = 1.0 + aa * d; if (Math.abs(d) < FPMIN) { d = FPMIN; } c = 1.0 + aa / c; if (Math.abs(c) < FPMIN) { c = FPMIN; } d = 1.0 / d; del = d * c; h *= del; i++; if (Math.abs(del - 1.0) < EPS) { test = false; } if (i > MAXITER) { test = false; System.err.println("Maximum number of iterations wes exceeded"); } } return h; } /** * Inverse of regularized incomplete beta function. */ public static double inverseRegularizedIncompleteBetaFunction(double alpha, double beta, double p) { final double EPS = 1.0E-8; double pp, t, u, err, x, al, h, w, afac; double a1 = alpha - 1.; double b1 = beta - 1.; if (p <= 0.0) { return 0.0; } if (p >= 1.0) { return 1.0; } if (alpha >= 1. && beta >= 1.) { pp = (p < 0.5) ? p : 1. - p; t = Math.sqrt(-2. * Math.log(pp)); x = (2.30753 + t * 0.27061) / (1. + t * (0.99229 + t * 0.04481)) - t; if (p < 0.5) { x = -x; } al = (x * x - 3.) / 6.; h = 2. / (1. / (2. * alpha - 1.) + 1. / (2. * beta - 1.)); w = (x * Math.sqrt(al + h) / h) - (1. / (2. * beta - 1) - 1. / (2. * alpha - 1.)) * (al + 5. / 6. - 2. / (3. * h)); x = alpha / (alpha + beta * Math.exp(2. * w)); } else { double lna = Math.log(alpha / (alpha + beta)); double lnb = Math.log(beta / (alpha + beta)); t = Math.exp(alpha * lna) / alpha; u = Math.exp(beta * lnb) / beta; w = t + u; if (p < t / w) { x = Math.pow(alpha * w * p, 1. / alpha); } else { x = 1. - Math.pow(beta * w * (1. - p), 1. / beta); } } afac = -Gamma.logGamma(alpha) - Gamma.logGamma(beta) + Gamma.logGamma(alpha + beta); for (int j = 0; j < 10; j++) { if (x == 0. || x == 1.) { return x; } err = regularizedIncompleteBetaFunction(alpha, beta, x) - p; t = Math.exp(a1 * Math.log(x) + b1 * Math.log(1. - x) + afac); u = err / t; x -= (t = u / (1. - 0.5 * Math.min(1., u * (a1 / x - b1 / (1. - x))))); if (x <= 0.) { x = 0.5 * (x + t); } if (x >= 1.) { x = 0.5 * (x + t + 1.); } if (Math.abs(t) < EPS * x && j > 0) { break; } } return x; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy