#ifndef SPROUT_RANDOM_BERNOULLI_DISTRIBUTION_HPP #define SPROUT_RANDOM_BERNOULLI_DISTRIBUTION_HPP #include #include #include #include namespace sprout { namespace random { // // bernoulli_distribution // template class bernoulli_distribution { public: typedef int input_type; typedef bool result_type; private: static SPROUT_CONSTEXPR bool arg_check_nothrow(RealType p_arg) { return p_arg >= 0 && p_arg <= 1; } static SPROUT_CONSTEXPR RealType arg_check(RealType p_arg) { return arg_check_nothrow(p_arg) ? p_arg : throw std::invalid_argument("bernoulli_distribution<>: invalid argument (p_arg >= 0 && p_arg <= 1)") ; } public: // // param_type // class param_type { public: typedef bernoulli_distribution distribution_type; private: static SPROUT_CONSTEXPR bool arg_check_nothrow(RealType p_arg) { return distribution_type::arg_check_nothrow(p_arg); } private: RealType p_; public: SPROUT_CONSTEXPR param_type() : p_(RealType(0.5)) {} SPROUT_CONSTEXPR explicit param_type(RealType p_arg) : p_(arg_check(p_arg)) {} SPROUT_CONSTEXPR RealType p() const { return p_; } template friend std::basic_istream& operator>>( std::basic_istream& lhs, param_type& rhs ) { RealType p; if (lhs >> p) { if (arg_check_nothrow(p)) { rhs.p_ = p; } else { lhs.setstate(std::ios_base::failbit); } } return lhs; } template friend std::basic_ostream& operator<<( std::basic_ostream& lhs, param_type const& rhs ) { return lhs << rhs.p_; } SPROUT_CONSTEXPR friend bool operator==(param_type const& lhs, param_type const& rhs) { return lhs.p_ == rhs.p_; } SPROUT_CONSTEXPR friend bool operator!=(param_type const& lhs, param_type const& rhs) { return !(lhs == rhs); } }; private: RealType p_; private: template SPROUT_CONSTEXPR sprout::random::random_result generate( sprout::random::random_result const& rnd ) const { return sprout::random::random_result( RealType(rnd.result() - rnd.engine().min()) <= p_ * RealType(rnd.engine().max() - rnd.engine().min()), rnd.engine(), *this ); } public: SPROUT_CONSTEXPR bernoulli_distribution() : p_(RealType(0.5)) {} SPROUT_CONSTEXPR explicit bernoulli_distribution(RealType p_arg) : p_(arg_check(p_arg)) {} SPROUT_CONSTEXPR explicit bernoulli_distribution(param_type const& parm) : p_(parm.p()) {} SPROUT_CONSTEXPR RealType p() const { return p_; } SPROUT_CONSTEXPR result_type min() const { return false; } SPROUT_CONSTEXPR result_type max() const { return true; } SPROUT_CONSTEXPR param_type param() const { return param_type(p_); } void param(param_type const& parm) { p_ = parm.p(); } template SPROUT_CONSTEXPR sprout::random::random_result operator()(Engine const& eng) const { return p_ == RealType(0) ? sprout::random::random_result(false, eng, *this) : generate(eng()) ; } template friend std::basic_istream& operator>>( std::basic_istream& lhs, bernoulli_distribution& rhs ) { param_type parm; if (lhs >> parm) { rhs.param(parm); } return lhs; } template friend std::basic_ostream& operator<<( std::basic_ostream& lhs, bernoulli_distribution const& rhs ) { return lhs << rhs.param(); } SPROUT_CONSTEXPR friend bool operator==(bernoulli_distribution const& lhs, bernoulli_distribution const& rhs) { return lhs.param() == rhs.param(); } SPROUT_CONSTEXPR friend bool operator!=(bernoulli_distribution const& lhs, bernoulli_distribution const& rhs) { return !(lhs == rhs); } }; } // namespace random using sprout::random::bernoulli_distribution; } // namespace sprout #endif // #ifndef SPROUT_RANDOM_BERNOULLI_DISTRIBUTION_HPP