363 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			363 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
|   | // Copyright John Maddock 2006, 2007. | ||
|  | // Copyright Paul A. Bristow 2007. | ||
|  | 
 | ||
|  | //  Use, modification and distribution are subject to the | ||
|  | //  Boost Software License, Version 1.0. (See accompanying file | ||
|  | //  LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) | ||
|  | 
 | ||
|  | #ifndef BOOST_STATS_CAUCHY_HPP | ||
|  | #define BOOST_STATS_CAUCHY_HPP | ||
|  | 
 | ||
|  | #ifdef _MSC_VER | ||
|  | #pragma warning(push) | ||
|  | #pragma warning(disable : 4127) // conditional expression is constant | ||
|  | #endif | ||
|  | 
 | ||
|  | #include <boost/math/distributions/fwd.hpp> | ||
|  | #include <boost/math/constants/constants.hpp> | ||
|  | #include <boost/math/distributions/complement.hpp> | ||
|  | #include <boost/math/distributions/detail/common_error_handling.hpp> | ||
|  | #include <boost/config/no_tr1/cmath.hpp> | ||
|  | 
 | ||
|  | #include <utility> | ||
|  | 
 | ||
|  | namespace boost{ namespace math | ||
|  | { | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | class cauchy_distribution; | ||
|  | 
 | ||
|  | namespace detail | ||
|  | { | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | RealType cdf_imp(const cauchy_distribution<RealType, Policy>& dist, const RealType& x, bool complement) | ||
|  | { | ||
|  |    // | ||
|  |    // This calculates the cdf of the Cauchy distribution and/or its complement. | ||
|  |    // | ||
|  |    // The usual formula for the Cauchy cdf is: | ||
|  |    // | ||
|  |    // cdf = 0.5 + atan(x)/pi | ||
|  |    // | ||
|  |    // But that suffers from cancellation error as x -> -INF. | ||
|  |    // | ||
|  |    // Recall that for x < 0: | ||
|  |    // | ||
|  |    // atan(x) = -pi/2 - atan(1/x) | ||
|  |    // | ||
|  |    // Substituting into the above we get: | ||
|  |    // | ||
|  |    // CDF = -atan(1/x)  ; x < 0 | ||
|  |    // | ||
|  |    // So the proceedure is to calculate the cdf for -fabs(x) | ||
|  |    // using the above formula, and then subtract from 1 when required | ||
|  |    // to get the result. | ||
|  |    // | ||
|  |    BOOST_MATH_STD_USING // for ADL of std functions | ||
|  |    static const char* function = "boost::math::cdf(cauchy<%1%>&, %1%)"; | ||
|  |    RealType result = 0; | ||
|  |    RealType location = dist.location(); | ||
|  |    RealType scale = dist.scale(); | ||
|  |    if(false == detail::check_location(function, location, &result, Policy())) | ||
|  |    { | ||
|  |      return result; | ||
|  |    } | ||
|  |    if(false == detail::check_scale(function, scale, &result, Policy())) | ||
|  |    { | ||
|  |       return result; | ||
|  |    } | ||
|  |    if(std::numeric_limits<RealType>::has_infinity && x == std::numeric_limits<RealType>::infinity()) | ||
|  |    { // cdf +infinity is unity. | ||
|  |      return static_cast<RealType>((complement) ? 0 : 1); | ||
|  |    } | ||
|  |    if(std::numeric_limits<RealType>::has_infinity && x == -std::numeric_limits<RealType>::infinity()) | ||
|  |    { // cdf -infinity is zero. | ||
|  |      return static_cast<RealType>((complement) ? 1 : 0); | ||
|  |    } | ||
|  |    if(false == detail::check_x(function, x, &result, Policy())) | ||
|  |    { // Catches x == NaN | ||
|  |       return result; | ||
|  |    } | ||
|  |    RealType mx = -fabs((x - location) / scale); // scale is > 0 | ||
|  |    if(mx > -tools::epsilon<RealType>() / 8) | ||
|  |    {  // special case first: x extremely close to location. | ||
|  |       return 0.5; | ||
|  |    } | ||
|  |    result = -atan(1 / mx) / constants::pi<RealType>(); | ||
|  |    return (((x > location) != complement) ? 1 - result : result); | ||
|  | } // cdf | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | RealType quantile_imp( | ||
|  |       const cauchy_distribution<RealType, Policy>& dist, | ||
|  |       const RealType& p, | ||
|  |       bool complement) | ||
|  | { | ||
|  |    // This routine implements the quantile for the Cauchy distribution, | ||
|  |    // the value p may be the probability, or its complement if complement=true. | ||
|  |    // | ||
|  |    // The procedure first performs argument reduction on p to avoid error | ||
|  |    // when calculating the tangent, then calulates the distance from the | ||
|  |    // mid-point of the distribution.  This is either added or subtracted | ||
|  |    // from the location parameter depending on whether `complement` is true. | ||
|  |    // | ||
|  |    static const char* function = "boost::math::quantile(cauchy<%1%>&, %1%)"; | ||
|  |    BOOST_MATH_STD_USING // for ADL of std functions | ||
|  | 
 | ||
|  |    RealType result = 0; | ||
|  |    RealType location = dist.location(); | ||
|  |    RealType scale = dist.scale(); | ||
|  |    if(false == detail::check_location(function, location, &result, Policy())) | ||
|  |    { | ||
|  |      return result; | ||
|  |    } | ||
|  |    if(false == detail::check_scale(function, scale, &result, Policy())) | ||
|  |    { | ||
|  |       return result; | ||
|  |    } | ||
|  |    if(false == detail::check_probability(function, p, &result, Policy())) | ||
|  |    { | ||
|  |       return result; | ||
|  |    } | ||
|  |    // Special cases: | ||
|  |    if(p == 1) | ||
|  |    { | ||
|  |       return (complement ? -1 : 1) * policies::raise_overflow_error<RealType>(function, 0, Policy()); | ||
|  |    } | ||
|  |    if(p == 0) | ||
|  |    { | ||
|  |       return (complement ? 1 : -1) * policies::raise_overflow_error<RealType>(function, 0, Policy()); | ||
|  |    } | ||
|  | 
 | ||
|  |    RealType P = p - floor(p);   // argument reduction of p: | ||
|  |    if(P > 0.5) | ||
|  |    { | ||
|  |       P = P - 1; | ||
|  |    } | ||
|  |    if(P == 0.5)   // special case: | ||
|  |    { | ||
|  |       return location; | ||
|  |    } | ||
|  |    result = -scale / tan(constants::pi<RealType>() * P); | ||
|  |    return complement ? RealType(location - result) : RealType(location + result); | ||
|  | } // quantile | ||
|  | 
 | ||
|  | } // namespace detail | ||
|  | 
 | ||
|  | template <class RealType = double, class Policy = policies::policy<> > | ||
|  | class cauchy_distribution | ||
|  | { | ||
|  | public: | ||
|  |    typedef RealType value_type; | ||
|  |    typedef Policy policy_type; | ||
|  | 
 | ||
|  |    cauchy_distribution(RealType l_location = 0, RealType l_scale = 1) | ||
|  |       : m_a(l_location), m_hg(l_scale) | ||
|  |    { | ||
|  |     static const char* function = "boost::math::cauchy_distribution<%1%>::cauchy_distribution"; | ||
|  |      RealType result; | ||
|  |      detail::check_location(function, l_location, &result, Policy()); | ||
|  |      detail::check_scale(function, l_scale, &result, Policy()); | ||
|  |    } // cauchy_distribution | ||
|  | 
 | ||
|  |    RealType location()const | ||
|  |    { | ||
|  |       return m_a; | ||
|  |    } | ||
|  |    RealType scale()const | ||
|  |    { | ||
|  |       return m_hg; | ||
|  |    } | ||
|  | 
 | ||
|  | private: | ||
|  |    RealType m_a;    // The location, this is the median of the distribution. | ||
|  |    RealType m_hg;   // The scale )or shape), this is the half width at half height. | ||
|  | }; | ||
|  | 
 | ||
|  | typedef cauchy_distribution<double> cauchy; | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline const std::pair<RealType, RealType> range(const cauchy_distribution<RealType, Policy>&) | ||
|  | { // Range of permissible values for random variable x. | ||
|  |   if (std::numeric_limits<RealType>::has_infinity) | ||
|  |   {  | ||
|  |      return std::pair<RealType, RealType>(-std::numeric_limits<RealType>::infinity(), std::numeric_limits<RealType>::infinity()); // - to + infinity. | ||
|  |   } | ||
|  |   else | ||
|  |   { // Can only use max_value. | ||
|  |    using boost::math::tools::max_value; | ||
|  |    return std::pair<RealType, RealType>(-max_value<RealType>(), max_value<RealType>()); // - to + max. | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline const std::pair<RealType, RealType> support(const cauchy_distribution<RealType, Policy>& ) | ||
|  | { // Range of supported values for random variable x. | ||
|  |    // This is range where cdf rises from 0 to 1, and outside it, the pdf is zero. | ||
|  |   if (std::numeric_limits<RealType>::has_infinity) | ||
|  |   {  | ||
|  |      return std::pair<RealType, RealType>(-std::numeric_limits<RealType>::infinity(), std::numeric_limits<RealType>::infinity()); // - to + infinity. | ||
|  |   } | ||
|  |   else | ||
|  |   { // Can only use max_value. | ||
|  |      using boost::math::tools::max_value; | ||
|  |      return std::pair<RealType, RealType>(-tools::max_value<RealType>(), max_value<RealType>()); // - to + max. | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType pdf(const cauchy_distribution<RealType, Policy>& dist, const RealType& x) | ||
|  | {   | ||
|  |    BOOST_MATH_STD_USING  // for ADL of std functions | ||
|  | 
 | ||
|  |    static const char* function = "boost::math::pdf(cauchy<%1%>&, %1%)"; | ||
|  |    RealType result = 0; | ||
|  |    RealType location = dist.location(); | ||
|  |    RealType scale = dist.scale(); | ||
|  |    if(false == detail::check_scale("boost::math::pdf(cauchy<%1%>&, %1%)", scale, &result, Policy())) | ||
|  |    { | ||
|  |       return result; | ||
|  |    } | ||
|  |    if(false == detail::check_location("boost::math::pdf(cauchy<%1%>&, %1%)", location, &result, Policy())) | ||
|  |    { | ||
|  |       return result; | ||
|  |    } | ||
|  |    if((boost::math::isinf)(x)) | ||
|  |    { | ||
|  |      return 0; // pdf + and - infinity is zero. | ||
|  |    } | ||
|  |    // These produce MSVC 4127 warnings, so the above used instead. | ||
|  |    //if(std::numeric_limits<RealType>::has_infinity && abs(x) == std::numeric_limits<RealType>::infinity()) | ||
|  |    //{ // pdf + and - infinity is zero. | ||
|  |    //  return 0; | ||
|  |    //} | ||
|  | 
 | ||
|  |    if(false == detail::check_x(function, x, &result, Policy())) | ||
|  |    { // Catches x = NaN | ||
|  |       return result; | ||
|  |    } | ||
|  | 
 | ||
|  |    RealType xs = (x - location) / scale; | ||
|  |    result = 1 / (constants::pi<RealType>() * scale * (1 + xs * xs)); | ||
|  |    return result; | ||
|  | } // pdf | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType cdf(const cauchy_distribution<RealType, Policy>& dist, const RealType& x) | ||
|  | { | ||
|  |    return detail::cdf_imp(dist, x, false); | ||
|  | } // cdf | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType quantile(const cauchy_distribution<RealType, Policy>& dist, const RealType& p) | ||
|  | { | ||
|  |    return detail::quantile_imp(dist, p, false); | ||
|  | } // quantile | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType cdf(const complemented2_type<cauchy_distribution<RealType, Policy>, RealType>& c) | ||
|  | { | ||
|  |    return detail::cdf_imp(c.dist, c.param, true); | ||
|  | } //  cdf complement | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType quantile(const complemented2_type<cauchy_distribution<RealType, Policy>, RealType>& c) | ||
|  | { | ||
|  |    return detail::quantile_imp(c.dist, c.param, true); | ||
|  | } // quantile complement | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType mean(const cauchy_distribution<RealType, Policy>&) | ||
|  | {  // There is no mean: | ||
|  |    typedef typename Policy::assert_undefined_type assert_type; | ||
|  |    BOOST_STATIC_ASSERT(assert_type::value == 0); | ||
|  | 
 | ||
|  |    return policies::raise_domain_error<RealType>( | ||
|  |       "boost::math::mean(cauchy<%1%>&)", | ||
|  |       "The Cauchy distribution does not have a mean: " | ||
|  |       "the only possible return value is %1%.", | ||
|  |       std::numeric_limits<RealType>::quiet_NaN(), Policy()); | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType variance(const cauchy_distribution<RealType, Policy>& /*dist*/) | ||
|  | { | ||
|  |    // There is no variance: | ||
|  |    typedef typename Policy::assert_undefined_type assert_type; | ||
|  |    BOOST_STATIC_ASSERT(assert_type::value == 0); | ||
|  | 
 | ||
|  |    return policies::raise_domain_error<RealType>( | ||
|  |       "boost::math::variance(cauchy<%1%>&)", | ||
|  |       "The Cauchy distribution does not have a variance: " | ||
|  |       "the only possible return value is %1%.", | ||
|  |       std::numeric_limits<RealType>::quiet_NaN(), Policy()); | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType mode(const cauchy_distribution<RealType, Policy>& dist) | ||
|  | { | ||
|  |    return dist.location(); | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType median(const cauchy_distribution<RealType, Policy>& dist) | ||
|  | { | ||
|  |    return dist.location(); | ||
|  | } | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType skewness(const cauchy_distribution<RealType, Policy>& /*dist*/) | ||
|  | { | ||
|  |    // There is no skewness: | ||
|  |    typedef typename Policy::assert_undefined_type assert_type; | ||
|  |    BOOST_STATIC_ASSERT(assert_type::value == 0); | ||
|  | 
 | ||
|  |    return policies::raise_domain_error<RealType>( | ||
|  |       "boost::math::skewness(cauchy<%1%>&)", | ||
|  |       "The Cauchy distribution does not have a skewness: " | ||
|  |       "the only possible return value is %1%.", | ||
|  |       std::numeric_limits<RealType>::quiet_NaN(), Policy()); // infinity? | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType kurtosis(const cauchy_distribution<RealType, Policy>& /*dist*/) | ||
|  | { | ||
|  |    // There is no kurtosis: | ||
|  |    typedef typename Policy::assert_undefined_type assert_type; | ||
|  |    BOOST_STATIC_ASSERT(assert_type::value == 0); | ||
|  | 
 | ||
|  |    return policies::raise_domain_error<RealType>( | ||
|  |       "boost::math::kurtosis(cauchy<%1%>&)", | ||
|  |       "The Cauchy distribution does not have a kurtosis: " | ||
|  |       "the only possible return value is %1%.", | ||
|  |       std::numeric_limits<RealType>::quiet_NaN(), Policy()); | ||
|  | } | ||
|  | 
 | ||
|  | template <class RealType, class Policy> | ||
|  | inline RealType kurtosis_excess(const cauchy_distribution<RealType, Policy>& /*dist*/) | ||
|  | { | ||
|  |    // There is no kurtosis excess: | ||
|  |    typedef typename Policy::assert_undefined_type assert_type; | ||
|  |    BOOST_STATIC_ASSERT(assert_type::value == 0); | ||
|  | 
 | ||
|  |    return policies::raise_domain_error<RealType>( | ||
|  |       "boost::math::kurtosis_excess(cauchy<%1%>&)", | ||
|  |       "The Cauchy distribution does not have a kurtosis: " | ||
|  |       "the only possible return value is %1%.", | ||
|  |       std::numeric_limits<RealType>::quiet_NaN(), Policy()); | ||
|  | } | ||
|  | 
 | ||
|  | } // namespace math | ||
|  | } // namespace boost | ||
|  | 
 | ||
|  | #ifdef _MSC_VER | ||
|  | #pragma warning(pop) | ||
|  | #endif | ||
|  | 
 | ||
|  | // This include must be at the end, *after* the accessors | ||
|  | // for this distribution have been defined, in order to | ||
|  | // keep compilers that support two-phase lookup happy. | ||
|  | #include <boost/math/distributions/detail/derived_accessors.hpp> | ||
|  | 
 | ||
|  | #endif // BOOST_STATS_CAUCHY_HPP |