对Boost库中的数值到字符串的转换的改进

类别:编程语言 点击:0 评论:0 推荐:

    Boost库中的lexical_cast函数功能很强大,但存在一个浮点数的精度问题。在阿炯的《走进Boost》中描述如下:

    #include <boost/lexical_cast.hpp>
    #include <string>
    #include <iostream>
   
    int main()
    {
        using std::string;
        const double d = 123.1234567;
        string s = boost::lexical_case<string>(d);
        std::cout<<s<<std::endl;
        return 0;
    }

    以上程序预想的结果是得到:"123.1234567",但是实际上我们只会得到"123.123",因为默认情况下std::stringstream的精度是6。这算是boost::lexical_cast的一个bug。

    阿炯提供一个解决办法:
    对头文件<boost/lexical_cast.hpp>作如下修改:
    //....
    template<typename Target, typename Source>
    Target lexical_cast(Source arg) {
     //.....
         Target result;
          //以上一行是增加内容
         interpreter.precision(std::numeric_limits<Source>::digits10);
         if(!(interpreter<<arg)||!(interpreter>>result)||!(interpreter>>std::ws).eof())
    //.........
    }

    我按照上述方法在我的机器上怎么也得不到正确结果。(BCB 6)
    就算可行的话,也存在以下问题:只能得到一种精度的字符串。

    我使用以下方法解决以上的问题:

    利用C++中的函数重载机制,增加一个函数:
    template<typename Target, typename Source>
    Target lexical_cast(Source arg)
    {
//......
        Target result;

         if(!(interpreter << arg) || !(interpreter >> result) ||
           !(interpreter >> std::ws).eof())
//....
    }

    template<typename Target, typename Source>
//  下面一行修改了
    Target lexical_cast(Source arg,int length)
    {
# ifdef BOOST_LEXICAL_CAST_USE_STRSTREAM
        std::strstream interpreter; // for out-of-the-box g++ 2.95.2
# else
        std::stringstream interpreter;
# endif
        Target result;
//      增加下面一行
        interpreter.precision( length );
        if(!(interpreter << arg) || !(interpreter >> result) ||
           !(interpreter >> std::ws).eof())
            throw bad_lexical_cast();

        return result;
    }
}

  使用时如下:
  std::string s = boost::lexical_cast<std::string>(123.1234567,10);
  即可以得到正确的结果,而且原来的函数不变。

小结

    不管多好的东西,总会有些小问题和不如意的东西,能够自己动手加以改进,是很好的事。

本文地址:http://com.8s8s.com/it/it29408.htm