mirror of
https://github.com/saitohirga/WSJT-X.git
synced 2025-10-24 01:20:22 -04:00
512 lines
19 KiB
Plaintext
512 lines
19 KiB
Plaintext
|
|
[section:pol_tutorial Policy Tutorial]
|
|
|
|
[section:what_is_a_policy So Just What is a Policy Anyway?]
|
|
|
|
A policy is a compile-time mechanism for customising the behaviour of a
|
|
special function, or a statistical distribution. With Policies you can
|
|
control:
|
|
|
|
* What action to take when an error occurs.
|
|
* What happens when you call a function that is mathematically undefined
|
|
(for example, if you ask for the mean of a Cauchy distribution).
|
|
* What happens when you ask for a quantile of a discrete distribution.
|
|
* Whether the library is allowed to internally promote `float` to `double`
|
|
and `double` to `long double` in order to improve precision.
|
|
* What precision to use when calculating the result.
|
|
|
|
Some of these policies could arguably be runtime variables, but then we couldn't
|
|
use compile-time dispatch internally to select the best evaluation method
|
|
for the given policies.
|
|
|
|
For this reason a Policy is a /type/: in fact it's an instance of the
|
|
class template `boost::math::policies::policy<>`. This class is just a
|
|
compile-time-container of user-selected policies (sometimes called a type-list):
|
|
|
|
using namespace boost::math::policies;
|
|
//
|
|
// Define a policy that sets ::errno on overflow, and does
|
|
// not promote double to long double internally:
|
|
//
|
|
typedef policy<domain_error<errno_on_error>, promote_double<false> > mypolicy;
|
|
|
|
[endsect] [/section:what_is_a_policy So Just What is a Policy Anyway?]
|
|
|
|
[section:policy_tut_defaults Policies Have Sensible Defaults]
|
|
|
|
Most of the time you can just ignore the policy framework.
|
|
|
|
['*The defaults for the various policies are as follows,
|
|
if these work OK for you then you can stop reading now!]
|
|
|
|
[variablelist
|
|
[[Domain Error][Throws a `std::domain_error` exception.]]
|
|
[[Pole Error][Occurs when a function is evaluated at a pole: throws a `std::domain_error` exception.]]
|
|
[[Overflow Error][Throws a `std::overflow_error` exception.]]
|
|
[[Underflow][Ignores the underflow, and returns zero.]]
|
|
[[Denormalised Result][Ignores the fact that the result is denormalised, and returns it.]]
|
|
[[Rounding Error][Throws a `boost::math::rounding_error` exception.]]
|
|
[[Internal Evaluation Error][Throws a `boost::math::evaluation_error` exception.]]
|
|
[[Indeterminate Result Error][Returns a result that depends on the function where the error occurred.]]
|
|
[[Promotion of float to double][Does occur by default - gives full float precision results.]]
|
|
[[Promotion of double to long double][Does occur by default if long double offers
|
|
more precision than double.]]
|
|
[[Precision of Approximation Used][By default uses an approximation that
|
|
will result in the lowest level of error for the type of the result.]]
|
|
[[Behaviour of Discrete Quantiles]
|
|
[
|
|
The quantile function will by default return an integer result that has been
|
|
/rounded outwards/. That is to say lower quantiles (where the probability is
|
|
less than 0.5) are rounded downward, and upper quantiles (where the probability
|
|
is greater than 0.5) are rounded upwards. This behaviour
|
|
ensures that if an X% quantile is requested, then /at least/ the requested
|
|
coverage will be present in the central region, and /no more than/
|
|
the requested coverage will be present in the tails.
|
|
|
|
This behaviour can be changed so that the quantile functions are rounded
|
|
differently, or even return a real-valued result using
|
|
[link math_toolkit.pol_overview Policies]. It is strongly
|
|
recommended that you read the tutorial
|
|
[link math_toolkit.pol_tutorial.understand_dis_quant
|
|
Understanding Quantiles of Discrete Distributions] before
|
|
using the quantile function on a discrete distribution. The
|
|
[link math_toolkit.pol_ref.discrete_quant_ref reference docs]
|
|
describe how to change the rounding policy
|
|
for these distributions.
|
|
]]
|
|
]
|
|
|
|
What's more, if you define your own policy type, then it automatically
|
|
inherits the defaults for any policies not explicitly set, so given:
|
|
|
|
using namespace boost::math::policies;
|
|
//
|
|
// Define a policy that sets ::errno on overflow, and does
|
|
// not promote double to long double internally:
|
|
//
|
|
typedef policy<domain_error<errno_on_error>, promote_double<false> > mypolicy;
|
|
|
|
then `mypolicy` defines a policy where only the overflow error handling and
|
|
`double`-promotion policies differ from the defaults.
|
|
|
|
[endsect][/section:policy_tut_defaults Policies Have Sensible Defaults]
|
|
|
|
[section:policy_usage So How are Policies Used Anyway?]
|
|
|
|
The details follow later, but basically policies can be set by either:
|
|
|
|
* Defining some macros that change the default behaviour: [*this is the
|
|
recommended method for setting installation-wide policies].
|
|
* By instantiating a distribution object with an explicit policy:
|
|
this is mainly reserved for ad hoc policy changes.
|
|
* By passing a policy to a special function as an optional final argument:
|
|
this is mainly reserved for ad hoc policy changes.
|
|
* By using some helper macros to define a set of functions or distributions
|
|
in the current namespace that use a specific policy: [*this is the
|
|
recommended method for setting policies on a project- or translation-unit-wide
|
|
basis].
|
|
|
|
The following sections introduce these methods in more detail.
|
|
|
|
[endsect] [/section:policy_usage So How are Policies Used Anyway?]
|
|
|
|
[section:changing_policy_defaults Changing the Policy Defaults]
|
|
|
|
The default policies used by the library are changed by the usual
|
|
configuration macro method.
|
|
|
|
For example, passing `-DBOOST_MATH_DOMAIN_ERROR_POLICY=errno_on_error` to
|
|
your compiler will cause domain errors to set `::errno` and return a __NaN
|
|
rather than the usual default behaviour of throwing a `std::domain_error`
|
|
exception.
|
|
|
|
[tip For Microsoft Visual Studio,you can add to the Project Property Page,
|
|
C/C++, Preprocessor, Preprocessor definitions like:
|
|
|
|
``BOOST_MATH_ASSERT_UNDEFINED_POLICY=0
|
|
BOOST_MATH_OVERFLOW_ERROR_POLICY=errno_on_error``
|
|
|
|
This may be helpful to avoid complications with pre-compiled headers
|
|
that may mean that the equivalent definitions in source code:
|
|
|
|
``#define BOOST_MATH_ASSERT_UNDEFINED_POLICY false
|
|
#define BOOST_MATH_OVERFLOW_ERROR_POLICY errno_on_error``
|
|
|
|
*may be ignored*.
|
|
|
|
The compiler command line shows:
|
|
|
|
``/D "BOOST_MATH_ASSERT_UNDEFINED_POLICY=0"
|
|
/D "BOOST_MATH_OVERFLOW_ERROR_POLICY=errno_on_error"``
|
|
] [/MSVC tip]
|
|
|
|
There is however a very important caveat to this:
|
|
|
|
[important
|
|
[*['Default policies changed by setting configuration macros must be changed
|
|
uniformly in every translation unit in the program.]]
|
|
|
|
Failure to follow this rule may result in violations of the "One
|
|
Definition Rule (ODR)" and result in unpredictable program behaviour.]
|
|
|
|
That means there are only two safe ways to use these macros:
|
|
|
|
* Edit them in [@../../../../boost/math/tools/user.hpp boost/math/tools/user.hpp],
|
|
so that the defaults are set on an installation-wide basis.
|
|
Unfortunately this may not be convenient if
|
|
you are using a pre-installed Boost distribution (on Linux for example).
|
|
* Set the defines in your project's Makefile or build environment, so that they
|
|
are set uniformly across all translation units.
|
|
|
|
What you should *not* do is:
|
|
|
|
* Set the defines in the source file using `#define` as doing so
|
|
almost certainly will break your program, unless you're absolutely
|
|
certain that the program is restricted to a single translation unit.
|
|
|
|
And, yes, you will find examples in our test programs where we break this
|
|
rule: but only because we know there will always be a single
|
|
translation unit only: ['don't say that you weren't warned!]
|
|
|
|
[import ../../example/error_handling_example.cpp]
|
|
|
|
[error_handling_example]
|
|
|
|
[endsect] [/section:changing_policy_defaults Changing the Policy Defaults]
|
|
|
|
[section:ad_hoc_dist_policies Setting Policies for Distributions on an Ad Hoc Basis]
|
|
|
|
All of the statistical distributions in this library are class templates
|
|
that accept two template parameters:
|
|
real type (float, double ...) and policy (how to handle exceptional events),
|
|
both with sensible defaults, for example:
|
|
|
|
namespace boost{ namespace math{
|
|
|
|
template <class RealType = double, class Policy = policies::policy<> >
|
|
class fisher_f_distribution;
|
|
|
|
typedef fisher_f_distribution<> fisher_f;
|
|
|
|
}}
|
|
|
|
This policy gets used by all the accessor functions that accept
|
|
a distribution as an argument, and forwarded to all the functions called
|
|
by these. So if you use the shorthand-typedef for the distribution, then you get
|
|
`double` precision arithmetic and all the default policies.
|
|
|
|
However, say for example we wanted to evaluate the quantile
|
|
of the binomial distribution at float precision, without internal
|
|
promotion to double, and with the result rounded to the /nearest/
|
|
integer, then here's how it can be done:
|
|
|
|
[import ../../example/policy_eg_3.cpp]
|
|
|
|
[policy_eg_3]
|
|
|
|
Which outputs:
|
|
|
|
[pre quantile is: 40]
|
|
|
|
[endsect][/section:ad_hoc_dist_policies Setting Policies for Distributions on an Ad Hoc Basis]
|
|
|
|
[section:ad_hoc_sf_policies Changing the Policy on an Ad Hoc Basis for the Special Functions]
|
|
|
|
All of the special functions in this library come in two overloaded forms,
|
|
one with a final "policy" parameter, and one without. For example:
|
|
|
|
namespace boost{ namespace math{
|
|
|
|
template <class RealType, class Policy>
|
|
RealType tgamma(RealType, const Policy&);
|
|
|
|
template <class RealType>
|
|
RealType tgamma(RealType);
|
|
|
|
}} // namespaces
|
|
|
|
Normally, the second version is just a forwarding wrapper to the first
|
|
like this:
|
|
|
|
template <class RealType>
|
|
inline RealType tgamma(RealType x)
|
|
{
|
|
return tgamma(x, policies::policy<>());
|
|
}
|
|
|
|
So calling a special function with a specific policy
|
|
is just a matter of defining the policy type to use
|
|
and passing it as the final parameter. For example,
|
|
suppose we want `tgamma` to behave in a C-compatible
|
|
fashion and set `::errno` when an error occurs, and never
|
|
throw an exception:
|
|
|
|
[import ../../example/policy_eg_1.cpp]
|
|
|
|
[policy_eg_1]
|
|
|
|
which outputs:
|
|
|
|
[pre
|
|
Result of tgamma(30000) is: 1.#INF
|
|
errno = 34
|
|
Result of tgamma(-10) is: 1.#QNAN
|
|
errno = 33
|
|
]
|
|
|
|
Alternatively, for ad hoc use, we can use the `make_policy`
|
|
helper function to create a policy for us: this usage is more
|
|
verbose, so is probably only preferred when a policy is going
|
|
to be used once only:
|
|
|
|
[import ../../example/policy_eg_2.cpp]
|
|
|
|
[policy_eg_2]
|
|
|
|
[endsect] [/section:ad_hoc_sf_policies Changing the Policy on an Ad Hoc Basis for the Special Functions]
|
|
|
|
[section:namespace_policies Setting Policies at Namespace or Translation Unit Scope]
|
|
|
|
Sometimes what you want to do is just change a set of policies within
|
|
the current scope: *the one thing you should not do in this situation
|
|
is use the configuration macros*, as this can lead to "One Definition
|
|
Rule" violations. Instead this library provides a pair of macros
|
|
especially for this purpose.
|
|
|
|
Let's consider the special functions first: we can declare a set of
|
|
forwarding functions that all use a specific policy using the
|
|
macro BOOST_MATH_DECLARE_SPECIAL_FUNCTIONS(['Policy]). This
|
|
macro should be used either inside a unique namespace set aside for the
|
|
purpose (for example, a C namespace for a C-style policy),
|
|
or an unnamed namespace if you just want the functions
|
|
visible in global scope for the current file only.
|
|
|
|
[import ../../example/policy_eg_4.cpp]
|
|
|
|
[policy_eg_4]
|
|
|
|
The same mechanism works well at file scope as well, by using an unnamed
|
|
namespace, we can ensure that these declarations don't conflict with any
|
|
alternate policies present in other translation units:
|
|
|
|
[import ../../example/policy_eg_5.cpp]
|
|
|
|
[policy_eg_5]
|
|
|
|
Handling policies for the statistical distributions is very similar except that now
|
|
the macro BOOST_MATH_DECLARE_DISTRIBUTIONS accepts two parameters: the
|
|
floating point type to use, and the policy type to apply. For example:
|
|
|
|
BOOST_MATH_DECLARE_DISTRIBUTIONS(double, mypolicy)
|
|
|
|
Results a set of typedefs being defined like this:
|
|
|
|
typedef boost::math::normal_distribution<double, mypolicy> normal;
|
|
|
|
The name of each typedef is the same as the name of the distribution
|
|
class template, but without the "_distribution" suffix.
|
|
|
|
[import ../../example/policy_eg_6.cpp]
|
|
|
|
[policy_eg_6]
|
|
|
|
[note
|
|
There is an important limitation to note: you can *not use the macros
|
|
BOOST_MATH_DECLARE_DISTRIBUTIONS and BOOST_MATH_DECLARE_SPECIAL_FUNCTIONS
|
|
['in the same namespace]*, as doing so creates ambiguities between functions
|
|
and distributions of the same name.
|
|
]
|
|
|
|
As before, the same mechanism works well at file scope as well: by using an unnamed
|
|
namespace, we can ensure that these declarations don't conflict with any
|
|
alternate policies present in other translation units:
|
|
|
|
[import ../../example/policy_eg_7.cpp]
|
|
|
|
[policy_eg_7]
|
|
|
|
[endsect][/section:namespace_policies Setting Policies at Namespace or Translation Unit Scope]
|
|
|
|
[section:user_def_err_pol Calling User Defined Error Handlers]
|
|
|
|
[import ../../example/policy_eg_8.cpp]
|
|
|
|
[policy_eg_8]
|
|
|
|
[import ../../example/policy_eg_9.cpp]
|
|
|
|
[policy_eg_9]
|
|
|
|
[endsect] [/section:user_def_err_pol Calling User Defined Error Handlers]
|
|
|
|
[section:understand_dis_quant Understanding Quantiles of Discrete Distributions]
|
|
|
|
Discrete distributions present us with a problem when calculating the
|
|
quantile: we are starting from a continuous real-valued variable - the
|
|
probability - but the result (the value of the random variable)
|
|
should really be discrete.
|
|
|
|
Consider for example a Binomial distribution, with a sample size of
|
|
50, and a success fraction of 0.5. There are a variety of ways
|
|
we can plot a discrete distribution, but if we plot the PDF
|
|
as a step-function then it looks something like this:
|
|
|
|
[$../graphs/binomial_pdf.png]
|
|
|
|
Now lets suppose that the user asks for a the quantile that corresponds
|
|
to a probability of 0.05, if we zoom in on the CDF for that region here's
|
|
what we see:
|
|
|
|
[$../graphs/binomial_quantile_1.png]
|
|
|
|
As can be seen there is no random variable that corresponds to
|
|
a probability of exactly 0.05, so we're left with two choices as
|
|
shown in the figure:
|
|
|
|
* We could round the result down to 18.
|
|
* We could round the result up to 19.
|
|
|
|
In fact there's actually a third choice as well: we could "pretend" that the
|
|
distribution was continuous and return a real valued result: in this case we
|
|
would calculate a result of approximately 18.701 (this accurately
|
|
reflects the fact that the result is nearer to 19 than 18).
|
|
|
|
By using policies we can offer any of the above as options, but that
|
|
still leaves the question: ['What is actually the right thing to do?]
|
|
|
|
And in particular: ['What policy should we use by default?]
|
|
|
|
In coming to an answer we should realise that:
|
|
|
|
* Calculating an integer result is often much faster than
|
|
calculating a real-valued result: in fact in our tests it
|
|
was up to 20 times faster.
|
|
* Normally people calculate quantiles so that they can perform
|
|
a test of some kind: ['"If the random variable is less than N
|
|
then we can reject our null-hypothesis with 90% confidence."]
|
|
|
|
So there is a genuine benefit to calculating an integer result
|
|
as well as it being "the right thing to do" from a philosophical
|
|
point of view. What's more if someone asks for a quantile at 0.05,
|
|
then we can normally assume that they are asking for
|
|
['[*at least] 95% of the probability to the right of the value chosen,
|
|
and [*no more than] 5% of the probability to the left of the value chosen.]
|
|
|
|
In the above binomial example we would therefore round the result down to 18.
|
|
|
|
The converse applies to upper-quantiles: If the probability is greater than
|
|
0.5 we would want to round the quantile up, ['so that [*at least] the requested
|
|
probability is to the left of the value returned, and [*no more than] 1 - the
|
|
requested probability is to the right of the value returned.]
|
|
|
|
Likewise for two-sided intervals, we would round lower quantiles down,
|
|
and upper quantiles up. This ensures that we have ['at least the requested
|
|
probability in the central region] and ['no more than 1 minus the requested
|
|
probability in the tail areas.]
|
|
|
|
For example, taking our 50 sample binomial distribution with a success fraction
|
|
of 0.5, if we wanted a two sided 90% confidence interval, then we would ask
|
|
for the 0.05 and 0.95 quantiles with the results ['rounded outwards] so that
|
|
['at least 90% of the probability] is in the central area:
|
|
|
|
[$../graphs/binomial_pdf_3.png]
|
|
|
|
So far so good, but there is in fact a trap waiting for the unwary here:
|
|
|
|
quantile(binomial(50, 0.5), 0.05);
|
|
|
|
returns 18 as the result, which is what we would expect from the graph above,
|
|
and indeed there is no x greater than 18 for which:
|
|
|
|
cdf(binomial(50, 0.5), x) <= 0.05;
|
|
|
|
However:
|
|
|
|
quantile(binomial(50, 0.5), 0.95);
|
|
|
|
returns 31, and indeed while there is no x less than 31 for which:
|
|
|
|
cdf(binomial(50, 0.5), x) >= 0.95;
|
|
|
|
We might naively expect that for this symmetrical distribution the result
|
|
would be 32 (since 32 = 50 - 18), but we need to remember that the cdf of
|
|
the binomial is /inclusive/ of the random variable. So while the left tail
|
|
area /includes/ the quantile returned, the right tail area always excludes
|
|
an upper quantile value: since that "belongs" to the central area.
|
|
|
|
Look at the graph above to see what's going on here: the lower quantile
|
|
of 18 belongs to the left tail, so any value <= 18 is in the left tail.
|
|
The upper quantile of 31 on the other hand belongs to the central area,
|
|
so the tail area actually starts at 32, so any value > 31 is in the
|
|
right tail.
|
|
|
|
Therefore if U and L are the upper and lower quantiles respectively, then
|
|
a random variable X is in the tail area - where we would reject the null
|
|
hypothesis if:
|
|
|
|
X <= L || X > U
|
|
|
|
And the a variable X is inside the central region if:
|
|
|
|
L < X <= U
|
|
|
|
The moral here is to ['always be very careful with your comparisons
|
|
when dealing with a discrete distribution], and if in doubt,
|
|
['base your comparisons on CDF's instead].
|
|
|
|
[heading Other Rounding Policies are Available]
|
|
|
|
As you would expect from a section on policies, you won't be surprised
|
|
to know that other rounding options are available:
|
|
|
|
[variablelist
|
|
|
|
[[integer_round_outwards]
|
|
[This is the default policy as described above: lower quantiles
|
|
are rounded down (probability < 0.5), and upper quantiles
|
|
(probability > 0.5) are rounded up.
|
|
|
|
This gives /no more than/ the requested probability
|
|
in the tails, and /at least/ the requested probability
|
|
in the central area.]]
|
|
[[integer_round_inwards]
|
|
[This is the exact opposite of the default policy:
|
|
lower quantiles
|
|
are rounded up (probability < 0.5),
|
|
and upper quantiles (probability > 0.5) are rounded down.
|
|
|
|
This gives /at least/ the requested probability
|
|
in the tails, and /no more than/ the requested probability
|
|
in the central area.]]
|
|
[[integer_round_down][This policy will always round the result down
|
|
no matter whether it is an upper or lower quantile]]
|
|
[[integer_round_up][This policy will always round the result up
|
|
no matter whether it is an upper or lower quantile]]
|
|
[[integer_round_nearest][This policy will always round the result
|
|
to the nearest integer
|
|
no matter whether it is an upper or lower quantile]]
|
|
[[real][This policy will return a real valued result
|
|
for the quantile of a discrete distribution: this is
|
|
generally much slower than finding an integer result
|
|
but does allow for more sophisticated rounding policies.]]
|
|
|
|
]
|
|
|
|
[import ../../example/policy_eg_10.cpp]
|
|
|
|
[policy_eg_10]
|
|
|
|
[endsect]
|
|
|
|
[endsect] [/section:pol_Tutorial Policy Tutorial]
|
|
|
|
|
|
[/ math.qbk
|
|
Copyright 2007, 2013 John Maddock and Paul A. Bristow.
|
|
Distributed under 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).
|
|
]
|
|
|
|
|