mirror of
https://github.com/bolero-MURAKAMI/Sprout
synced 2024-11-12 21:09:01 +00:00
320 lines
9.6 KiB
C++
320 lines
9.6 KiB
C++
#ifndef SPROUT_RANDOM_RANDOM_RESULT_HPP
|
|
#define SPROUT_RANDOM_RANDOM_RESULT_HPP
|
|
|
|
#include <cstddef>
|
|
#include <iterator>
|
|
#include <utility>
|
|
#include <type_traits>
|
|
#include <sprout/config.hpp>
|
|
#include <sprout/iterator/next.hpp>
|
|
#include <sprout/utility/swap.hpp>
|
|
|
|
namespace sprout {
|
|
namespace random {
|
|
//
|
|
// random_result
|
|
//
|
|
template<typename Engine, typename Distribution = void, typename Enable = void>
|
|
class random_result;
|
|
|
|
template<typename Engine, typename Distribution>
|
|
class random_result<
|
|
Engine,
|
|
Distribution,
|
|
typename std::enable_if<!std::is_same<Distribution, void>::value>::type
|
|
>
|
|
: public std::iterator<
|
|
std::input_iterator_tag,
|
|
typename Distribution::result_type,
|
|
std::ptrdiff_t,
|
|
typename Distribution::result_type const*,
|
|
typename Distribution::result_type const&
|
|
>
|
|
{
|
|
public:
|
|
typedef Engine engine_type;
|
|
typedef Distribution distribution_type;
|
|
typedef typename distribution_type::result_type result_type;
|
|
private:
|
|
typedef std::iterator<
|
|
std::input_iterator_tag,
|
|
result_type,
|
|
std::ptrdiff_t,
|
|
result_type const*,
|
|
result_type const&
|
|
> base_type;
|
|
public:
|
|
typedef typename base_type::iterator_category iterator_category;
|
|
typedef typename base_type::value_type value_type;
|
|
typedef typename base_type::difference_type difference_type;
|
|
typedef typename base_type::pointer pointer;
|
|
typedef typename base_type::reference reference;
|
|
private:
|
|
result_type result_;
|
|
engine_type engine_;
|
|
distribution_type distribution_;
|
|
public:
|
|
SPROUT_CONSTEXPR random_result()
|
|
: result_()
|
|
, engine_()
|
|
, distribution_()
|
|
{}
|
|
SPROUT_CONSTEXPR random_result(
|
|
result_type result,
|
|
engine_type const& engine,
|
|
distribution_type const& distribution
|
|
)
|
|
: result_(result)
|
|
, engine_(engine)
|
|
, distribution_(distribution)
|
|
{}
|
|
SPROUT_CONSTEXPR operator result_type() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR random_result operator()() const {
|
|
return distribution_(engine_);
|
|
}
|
|
result_type& result() SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR result_type const& result() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
result_type& generated_value() SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR result_type const& generated_value() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
engine_type& engine() SPROUT_NOEXCEPT {
|
|
return engine_;
|
|
}
|
|
SPROUT_CONSTEXPR engine_type const& engine() const SPROUT_NOEXCEPT {
|
|
return engine_;
|
|
}
|
|
random_result& next_generator() SPROUT_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
SPROUT_CONSTEXPR random_result const& next_generator() const SPROUT_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
distribution_type& distribution() SPROUT_NOEXCEPT {
|
|
return distribution_;
|
|
}
|
|
SPROUT_CONSTEXPR distribution_type const& distribution() const SPROUT_NOEXCEPT {
|
|
return distribution_;
|
|
}
|
|
SPROUT_CONSTEXPR result_type min() const SPROUT_NOEXCEPT {
|
|
return distribution_.min();
|
|
}
|
|
SPROUT_CONSTEXPR result_type max() const SPROUT_NOEXCEPT {
|
|
return distribution_.max();
|
|
}
|
|
void swap(random_result& other)
|
|
SPROUT_NOEXCEPT_EXPR(
|
|
SPROUT_NOEXCEPT_EXPR(sprout::swap(result_, other.result_))
|
|
&& SPROUT_NOEXCEPT_EXPR(sprout::swap(engine_, other.engine_))
|
|
&& SPROUT_NOEXCEPT_EXPR(sprout::swap(distribution_, other.distribution_))
|
|
)
|
|
{
|
|
sprout::swap(result_, other.result_);
|
|
sprout::swap(engine_, other.engine_);
|
|
sprout::swap(distribution_, other.distribution_);
|
|
}
|
|
friend SPROUT_CONSTEXPR bool operator==(random_result const& lhs, random_result const& rhs) SPROUT_NOEXCEPT {
|
|
return lhs.result_ == rhs.result_
|
|
&& lhs.engine_ == rhs.engine_
|
|
&& lhs.distribution_ == rhs.distribution_
|
|
;
|
|
}
|
|
friend SPROUT_CONSTEXPR bool operator!=(random_result const& lhs, random_result const& rhs) SPROUT_NOEXCEPT {
|
|
return !(lhs == rhs);
|
|
}
|
|
SPROUT_CONSTEXPR reference operator*() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR pointer operator->() const SPROUT_NOEXCEPT {
|
|
return &result_;
|
|
}
|
|
random_result& operator++() {
|
|
random_result temp((*this)());
|
|
temp.swap(*this);
|
|
return *this;
|
|
}
|
|
random_result operator++(int) {
|
|
random_result result(*this);
|
|
++*this;
|
|
return result;
|
|
}
|
|
};
|
|
template<typename Engine, typename Distribution>
|
|
class random_result<
|
|
Engine,
|
|
Distribution,
|
|
typename std::enable_if<std::is_same<Distribution, void>::value>::type
|
|
>
|
|
: public std::iterator<
|
|
std::input_iterator_tag,
|
|
typename Engine::result_type,
|
|
std::ptrdiff_t,
|
|
typename Engine::result_type const*,
|
|
typename Engine::result_type const&
|
|
>
|
|
{
|
|
public:
|
|
typedef Engine engine_type;
|
|
typedef typename engine_type::result_type result_type;
|
|
private:
|
|
typedef std::iterator<
|
|
std::input_iterator_tag,
|
|
result_type,
|
|
std::ptrdiff_t,
|
|
result_type const*,
|
|
result_type const&
|
|
> base_type;
|
|
public:
|
|
typedef typename base_type::iterator_category iterator_category;
|
|
typedef typename base_type::value_type value_type;
|
|
typedef typename base_type::difference_type difference_type;
|
|
typedef typename base_type::pointer pointer;
|
|
typedef typename base_type::reference reference;
|
|
private:
|
|
result_type result_;
|
|
engine_type engine_;
|
|
public:
|
|
SPROUT_CONSTEXPR random_result()
|
|
: result_()
|
|
, engine_()
|
|
{}
|
|
SPROUT_CONSTEXPR random_result(
|
|
result_type result,
|
|
engine_type const& engine
|
|
)
|
|
: result_(result)
|
|
, engine_(engine)
|
|
{}
|
|
SPROUT_CONSTEXPR operator result_type() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR random_result operator()() const {
|
|
return engine_();
|
|
}
|
|
result_type& result() SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR result_type const& result() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
result_type& generated_value() SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR result_type const& generated_value() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
engine_type& engine() SPROUT_NOEXCEPT {
|
|
return engine_;
|
|
}
|
|
SPROUT_CONSTEXPR engine_type const& engine() const SPROUT_NOEXCEPT {
|
|
return engine_;
|
|
}
|
|
random_result& next_generator() SPROUT_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
SPROUT_CONSTEXPR random_result const& next_generator() const SPROUT_NOEXCEPT {
|
|
return *this;
|
|
}
|
|
SPROUT_CONSTEXPR result_type min() const SPROUT_NOEXCEPT {
|
|
return engine_.min();
|
|
}
|
|
SPROUT_CONSTEXPR result_type max() const SPROUT_NOEXCEPT {
|
|
return engine_.max();
|
|
}
|
|
friend SPROUT_CONSTEXPR bool operator==(random_result const& lhs, random_result const& rhs) SPROUT_NOEXCEPT {
|
|
return lhs.result_ == rhs.result_
|
|
&& lhs.engine_ == rhs.engine_
|
|
;
|
|
}
|
|
friend SPROUT_CONSTEXPR bool operator!=(random_result const& lhs, random_result const& rhs) SPROUT_NOEXCEPT {
|
|
return !(lhs == rhs);
|
|
}
|
|
void swap(random_result& other)
|
|
SPROUT_NOEXCEPT_EXPR(
|
|
SPROUT_NOEXCEPT_EXPR(sprout::swap(result_, other.result_))
|
|
&& SPROUT_NOEXCEPT_EXPR(sprout::swap(engine_, other.engine_))
|
|
)
|
|
{
|
|
sprout::swap(result_, other.result_);
|
|
sprout::swap(engine_, other.engine_);
|
|
}
|
|
SPROUT_CONSTEXPR reference operator*() const SPROUT_NOEXCEPT {
|
|
return result_;
|
|
}
|
|
SPROUT_CONSTEXPR pointer operator->() const SPROUT_NOEXCEPT {
|
|
return &result_;
|
|
}
|
|
random_result& operator++() {
|
|
random_result temp((*this)());
|
|
temp.swap(*this);
|
|
return *this;
|
|
}
|
|
random_result operator++(int) {
|
|
random_result result(*this);
|
|
random_result temp((*this)());
|
|
temp.swap(*this);
|
|
return result;
|
|
}
|
|
};
|
|
|
|
//
|
|
// swap
|
|
//
|
|
template<typename Engine, typename Distribution>
|
|
inline void
|
|
swap(sprout::random::random_result<Engine, Distribution>& lhs, sprout::random::random_result<Engine, Distribution>& rhs)
|
|
SPROUT_NOEXCEPT_EXPR(SPROUT_NOEXCEPT_EXPR(lhs.swap(rhs)))
|
|
{
|
|
lhs.swap(rhs);
|
|
}
|
|
|
|
//
|
|
// iterator_next
|
|
//
|
|
template<typename Engine, typename Distribution>
|
|
inline SPROUT_CONSTEXPR sprout::random::random_result<Engine, Distribution>
|
|
iterator_next(sprout::random::random_result<Engine, Distribution> const& it) {
|
|
return it();
|
|
}
|
|
|
|
//
|
|
// generated_value
|
|
//
|
|
template<typename Engine, typename Distribution>
|
|
inline typename sprout::random::random_result<Engine, Distribution>::result_type&
|
|
generated_value(sprout::random::random_result<Engine, Distribution>& t) {
|
|
return t.generated_value();
|
|
}
|
|
template<typename Engine, typename Distribution>
|
|
inline SPROUT_CONSTEXPR typename sprout::random::random_result<Engine, Distribution>::result_type const&
|
|
generated_value(sprout::random::random_result<Engine, Distribution> const& t) {
|
|
return t.generated_value();
|
|
}
|
|
|
|
//
|
|
// next_generator
|
|
//
|
|
template<typename Engine, typename Distribution>
|
|
inline typename sprout::random::random_result<Engine, Distribution>&
|
|
next_generator(sprout::random::random_result<Engine, Distribution>& t) {
|
|
return t.next_generator();
|
|
}
|
|
template<typename Engine, typename Distribution>
|
|
inline SPROUT_CONSTEXPR typename sprout::random::random_result<Engine, Distribution> const&
|
|
next_generator(sprout::random::random_result<Engine, Distribution> const& t) {
|
|
return t.next_generator();
|
|
}
|
|
} // namespace random
|
|
|
|
using sprout::random::random_result;
|
|
} // namespace sprout
|
|
|
|
#endif // #ifndef SPROUT_RANDOM_RANDOM_RESULT_HPP
|