real.cpp
来自「Boost provides free peer-reviewed portab」· C++ 代码 · 共 375 行
CPP
375 行
/*============================================================================= Copyright (c) 2001-2007 Joel de Guzman Copyright (c) 2001-2008 Hartmut Kaiser Use, modification and distribution is 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)=============================================================================*/#include <climits>#include <boost/math/concepts/real_concept.hpp>#include <boost/detail/lightweight_test.hpp>#include <boost/spirit/include/qi_char.hpp>#include <boost/spirit/include/qi_numeric.hpp>#include <boost/spirit/include/qi_operator.hpp>#include <boost/spirit/home/support/detail/math/fpclassify.hpp>#include <boost/spirit/home/support/detail/math/signbit.hpp>#include "test.hpp"using namespace spirit_test;///////////////////////////////////////////////////////////////////////////////// These policies can be used to parse thousand separated// numbers with at most 2 decimal digits after the decimal// point. e.g. 123,456,789.01///////////////////////////////////////////////////////////////////////////////template <typename T>struct ts_real_policies : boost::spirit::qi::ureal_policies<T>{ // 2 decimal places Max template <typename Iterator, typename Attribute> static bool parse_frac_n(Iterator& first, Iterator const& last, Attribute& attr) { return boost::spirit::qi:: extract_uint<T, 10, 1, 2, true>::call(first, last, attr); } // No exponent template <typename Iterator> static bool parse_exp(Iterator&, Iterator const&) { return false; } // No exponent template <typename Iterator, typename Attribute> static bool parse_exp_n(Iterator&, Iterator const&, Attribute&) { return false; } // Thousands separated numbers template <typename Iterator, typename Attribute> static bool parse_n(Iterator& first, Iterator const& last, Attribute& attr) { using namespace boost::spirit::qi; using namespace boost::spirit; uint_spec<unsigned, 10, 1, 3> uint3; uint_spec<unsigned, 10, 3, 3> uint3_3; T result = 0; if (parse(first, last, uint3, result)) { bool hit = false; T n; Iterator save = first; while (parse(first, last, ',') && parse(first, last, uint3_3, n)) { result = result * 1000 + n; save = first; hit = true; } first = save; if (hit) attr = result; return hit; } return false; }};template <typename T>struct no_trailing_dot_policy : boost::spirit::qi::real_policies<T>{ static bool const allow_trailing_dot = false;};template <typename T>struct no_leading_dot_policy : boost::spirit::qi::real_policies<T>{ static bool const allow_leading_dot = false;};template <typename T>boolcompare(T n, double expected){ double const eps = 0.00001; T delta = n - expected; return (delta >= -eps) && (delta <= eps);}intmain(){ /////////////////////////////////////////////////////////////////////////////// // thousand separated numbers /////////////////////////////////////////////////////////////////////////////// { using boost::spirit::qi::uint_spec; using boost::spirit::qi::parse; uint_spec<unsigned, 10, 1, 3> uint3; uint_spec<unsigned, 10, 3, 3> uint3_3; #define r (uint3 >> *(',' >> uint3_3)) BOOST_TEST(test("1,234,567,890", r)); BOOST_TEST(test("12,345,678,900", r)); BOOST_TEST(test("123,456,789,000", r)); BOOST_TEST(!test("1000,234,567,890", r)); BOOST_TEST(!test("1,234,56,890", r)); BOOST_TEST(!test("1,66", r)); } /////////////////////////////////////////////////////////////////////////////// // unsigned real number tests /////////////////////////////////////////////////////////////////////////////// { using boost::spirit::qi::real_spec; using boost::spirit::qi::parse; using boost::spirit::qi::ureal_policies; real_spec<double, ureal_policies<double> > udouble; double d; BOOST_TEST(test("1234", udouble)); BOOST_TEST(test_attr("1234", udouble, d) && compare(d, 1234)); BOOST_TEST(test("1.2e3", udouble)); BOOST_TEST(test_attr("1.2e3", udouble, d) && compare(d, 1.2e3)); BOOST_TEST(test("1.2e-3", udouble)); BOOST_TEST(test_attr("1.2e-3", udouble, d) && compare(d, 1.2e-3)); BOOST_TEST(test("1.e2", udouble)); BOOST_TEST(test_attr("1.e2", udouble, d) && compare(d, 1.e2)); BOOST_TEST(test("1.", udouble)); BOOST_TEST(test_attr("1.", udouble, d) && compare(d, 1.)); BOOST_TEST(test(".2e3", udouble)); BOOST_TEST(test_attr(".2e3", udouble, d) && compare(d, .2e3)); BOOST_TEST(test("2e3", udouble)); BOOST_TEST(test_attr("2e3", udouble, d) && compare(d, 2e3)); BOOST_TEST(test("2", udouble)); BOOST_TEST(test_attr("2", udouble, d) && compare(d, 2)); using boost::math::fpclassify; BOOST_TEST(test("inf", udouble)); BOOST_TEST(test("infinity", udouble)); BOOST_TEST(test("INF", udouble)); BOOST_TEST(test("INFINITY", udouble)); BOOST_TEST(test_attr("inf", udouble, d) && FP_INFINITE == fpclassify(d)); BOOST_TEST(test_attr("INF", udouble, d) && FP_INFINITE == fpclassify(d)); BOOST_TEST(test_attr("infinity", udouble, d) && FP_INFINITE == fpclassify(d)); BOOST_TEST(test_attr("INFINITY", udouble, d) && FP_INFINITE == fpclassify(d)); BOOST_TEST(test("nan", udouble)); BOOST_TEST(test_attr("nan", udouble, d) && FP_NAN == fpclassify(d)); BOOST_TEST(test("NAN", udouble)); BOOST_TEST(test_attr("NAN", udouble, d) && FP_NAN == fpclassify(d)); BOOST_TEST(test("nan(...)", udouble)); BOOST_TEST(test_attr("nan(...)", udouble, d) && FP_NAN == fpclassify(d)); BOOST_TEST(test("NAN(...)", udouble)); BOOST_TEST(test_attr("NAN(...)", udouble, d) && FP_NAN == fpclassify(d)); BOOST_TEST(!test("e3", udouble)); BOOST_TEST(!test_attr("e3", udouble, d)); BOOST_TEST(!test("-1.2e3", udouble)); BOOST_TEST(!test_attr("-1.2e3", udouble, d)); BOOST_TEST(!test("+1.2e3", udouble)); BOOST_TEST(!test_attr("+1.2e3", udouble, d)); BOOST_TEST(!test("1.2e", udouble)); BOOST_TEST(!test_attr("1.2e", udouble, d)); BOOST_TEST(!test("-.3", udouble)); BOOST_TEST(!test_attr("-.3", udouble, d)); }///////////////////////////////////////////////////////////////////////////////// signed real number tests/////////////////////////////////////////////////////////////////////////////// { using boost::spirit::double_; using boost::spirit::qi::parse; double d; BOOST_TEST(test("-1234", double_)); BOOST_TEST(test_attr("-1234", double_, d) && compare(d, -1234)); BOOST_TEST(test("-1.2e3", double_)); BOOST_TEST(test_attr("-1.2e3", double_, d) && compare(d, -1.2e3)); BOOST_TEST(test("+1.2e3", double_)); BOOST_TEST(test_attr("+1.2e3", double_, d) && compare(d, 1.2e3)); BOOST_TEST(test("-0.1", double_)); BOOST_TEST(test_attr("-0.1", double_, d) && compare(d, -0.1)); BOOST_TEST(test("-1.2e-3", double_)); BOOST_TEST(test_attr("-1.2e-3", double_, d) && compare(d, -1.2e-3)); BOOST_TEST(test("-1.e2", double_)); BOOST_TEST(test_attr("-1.e2", double_, d) && compare(d, -1.e2)); BOOST_TEST(test("-.2e3", double_)); BOOST_TEST(test_attr("-.2e3", double_, d) && compare(d, -.2e3)); BOOST_TEST(test("-2e3", double_)); BOOST_TEST(test_attr("-2e3", double_, d) && compare(d, -2e3)); BOOST_TEST(!test("-e3", double_)); BOOST_TEST(!test_attr("-e3", double_, d)); BOOST_TEST(!test("-1.2e", double_)); BOOST_TEST(!test_attr("-1.2e", double_, d)); using boost::math::fpclassify; using boost::math::signbit; BOOST_TEST(test("-inf", double_)); BOOST_TEST(test("-infinity", double_)); BOOST_TEST(test_attr("-inf", double_, d) && FP_INFINITE == fpclassify(d) && signbit(d)); BOOST_TEST(test_attr("-infinity", double_, d) && FP_INFINITE == fpclassify(d) && signbit(d)); BOOST_TEST(test("-INF", double_)); BOOST_TEST(test("-INFINITY", double_)); BOOST_TEST(test_attr("-INF", double_, d) && FP_INFINITE == fpclassify(d) && signbit(d)); BOOST_TEST(test_attr("-INFINITY", double_, d) && FP_INFINITE == fpclassify(d) && signbit(d)); BOOST_TEST(test("-nan", double_)); BOOST_TEST(test_attr("-nan", double_, d) && FP_NAN == fpclassify(d) && signbit(d)); BOOST_TEST(test("-NAN", double_)); BOOST_TEST(test_attr("-NAN", double_, d) && FP_NAN == fpclassify(d) && signbit(d)); BOOST_TEST(test("-nan(...)", double_)); BOOST_TEST(test_attr("-nan(...)", double_, d) && FP_NAN == fpclassify(d) && signbit(d)); BOOST_TEST(test("-NAN(...)", double_)); BOOST_TEST(test_attr("-NAN(...)", double_, d) && FP_NAN == fpclassify(d) && signbit(d)); } /////////////////////////////////////////////////////////////////////////////// // strict real number tests /////////////////////////////////////////////////////////////////////////////// { using boost::spirit::qi::real_spec; using boost::spirit::qi::parse; using boost::spirit::qi::strict_ureal_policies; using boost::spirit::qi::strict_real_policies; real_spec<double, strict_ureal_policies<double> > strict_udouble; real_spec<double, strict_real_policies<double> > strict_double; double d; BOOST_TEST(!test("1234", strict_udouble)); BOOST_TEST(!test_attr("1234", strict_udouble, d)); BOOST_TEST(test("1.2", strict_udouble)); BOOST_TEST(test_attr("1.2", strict_udouble, d) && compare(d, 1.2)); BOOST_TEST(!test("-1234", strict_double)); BOOST_TEST(!test_attr("-1234", strict_double, d)); BOOST_TEST(test("123.", strict_double)); BOOST_TEST(test_attr("123.", strict_double, d) && compare(d, 123)); BOOST_TEST(test("3.E6", strict_double)); BOOST_TEST(test_attr("3.E6", strict_double, d) && compare(d, 3e6)); real_spec<double, no_trailing_dot_policy<double> > notrdot_real; real_spec<double, no_leading_dot_policy<double> > nolddot_real; BOOST_TEST(!test("1234.", notrdot_real)); // Bad trailing dot BOOST_TEST(!test(".1234", nolddot_real)); // Bad leading dot } /////////////////////////////////////////////////////////////////////////// // Special thousands separated numbers /////////////////////////////////////////////////////////////////////////// { using boost::spirit::qi::real_spec; using boost::spirit::qi::parse; real_spec<double, ts_real_policies<double> > ts_real; double d; BOOST_TEST(test("123,456,789.01", ts_real)); BOOST_TEST(test_attr("123,456,789.01", ts_real, d) && compare(d, 123456789.01)); BOOST_TEST(test("12,345,678.90", ts_real)); BOOST_TEST(test_attr("12,345,678.90", ts_real, d) && compare(d, 12345678.90)); BOOST_TEST(test("1,234,567.89", ts_real)); BOOST_TEST(test_attr("1,234,567.89", ts_real, d) && compare(d, 1234567.89)); BOOST_TEST(!test("1234,567,890", ts_real)); BOOST_TEST(!test("1,234,5678,9", ts_real)); BOOST_TEST(!test("1,234,567.89e6", ts_real)); BOOST_TEST(!test("1,66", ts_real)); } /////////////////////////////////////////////////////////////////////////// // Custom data type /////////////////////////////////////////////////////////////////////////// { using boost::math::concepts::real_concept; using boost::spirit::qi::real_spec; using boost::spirit::qi::real_policies; using boost::spirit::qi::parse; real_spec<real_concept, real_policies<real_concept> > custom_real; real_concept d; BOOST_TEST(test("-1234", custom_real)); BOOST_TEST(test_attr("-1234", custom_real, d) && compare(d, -1234)); BOOST_TEST(test("-1.2e3", custom_real)); BOOST_TEST(test_attr("-1.2e3", custom_real, d) && compare(d, -1.2e3)); BOOST_TEST(test("+1.2e3", custom_real)); BOOST_TEST(test_attr("+1.2e3", custom_real, d) && compare(d, 1.2e3)); BOOST_TEST(test("-0.1", custom_real)); BOOST_TEST(test_attr("-0.1", custom_real, d) && compare(d, -0.1)); BOOST_TEST(test("-1.2e-3", custom_real)); BOOST_TEST(test_attr("-1.2e-3", custom_real, d) && compare(d, -1.2e-3)); BOOST_TEST(test("-1.e2", custom_real)); BOOST_TEST(test_attr("-1.e2", custom_real, d) && compare(d, -1.e2)); BOOST_TEST(test("-.2e3", custom_real)); BOOST_TEST(test_attr("-.2e3", custom_real, d) && compare(d, -.2e3)); BOOST_TEST(test("-2e3", custom_real)); BOOST_TEST(test_attr("-2e3", custom_real, d) && compare(d, -2e3)); BOOST_TEST(!test("-e3", custom_real)); BOOST_TEST(!test_attr("-e3", custom_real, d)); BOOST_TEST(!test("-1.2e", custom_real)); BOOST_TEST(!test_attr("-1.2e", custom_real, d)); } return boost::report_errors();}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?