boost属性树put / get DBL_MAX

时间:2012-11-30 20:46:50

标签: c++ boost boost-propertytree

我正在编写一个ptree,在某些时候我需要将DBL_MAX放入(作为默认值)。 当我打开生成的xml文件时,我看到了正确的数字。

但是当我使用ptree.get获取数字时会抛出异常:conversion of data to type "d" failed

这是我的代码:

using boost::property_tree::ptree;
ptree pt;

double d=-DBL_MAX;
double d2=-1.797693134862316e+308;
double d3=-1.79769e+308;

cout<<d<<endl;
cout<<d2<<endl;
cout<<d3<<endl;

pt.put<double>("double", d);
write_xml("test.xml", pt);

cout << "totalEndTimeLowerBound: " << pt.get<double>("double")<<endl;
//doesn't work with d and d2, but works with d3

导致此错误的原因是什么?如何解决?

1 个答案:

答案 0 :(得分:2)

默认情况下,ptree将其值存储为std::string,并使用basic_stringstream精确转换它们:

s.precision(std::numeric_limits<double>::digits10+1);

DBL_MAX转换为std::string时会出现此问题,因为它会将数字四舍五入为无效值。您可以使用以下代码进行检查:

ptree pt;
pt.put("min", -DBL_MAX);
pt.put("max", DBL_MAX);
cout << "Min=" << pt.get<string>("min") << std::endl;
cout << "Max=" << pt.get<string>("max") << std::endl;

使用Visual Studio打印:

  

最小= -1.797693134862316e + 308

     

最大= 1.797693134862316e + 308

但是,DBL_MAX定义为1.7976931348623158e+308,因此打印值超出限制。

有几种解决方法,但没有一种是完美的:

  • 使用小于DBL_MAX的其他默认值。例如1.797693134862315e+308
  • 抓住bad_data例外,并假设它意味着默认。
  • 使用自定义转换器注册新类型。您可以看到示例here
  • 降低存储值的精度。您可以使用以下代码执行此操作:

    namespace boost { namespace property_tree 
    {
      template <typename Ch, typename Traits>
      struct customize_stream<Ch, Traits, double, void>
      {
        static void insert(std::basic_ostream<Ch, Traits>& s, const double& e) {
          s.precision(std::numeric_limits<double>::digits10-1);
          s << e;
        }
        static void extract(std::basic_istream<Ch, Traits>& s, double& e) {
          s >> e;
          if(!s.eof()) {
            s >> std::ws;
           }
        }
      };
    }