我正在对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

什么会导致此错误,我该如何解决?

最佳答案

默认情况下,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,它可以打印:



但是,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;
           }
        }
      };
    }
    
  • 10-08 08:55