4

我必须使用非科学(即没有尾数/指数/E)字符串转换十进制数。我的代码如下所示:

/*!
\brief Converts a <em>XML Schema Decimal</em>
*/
char *ToDecimal(double val) const
{
    const size_t nMax = 200;
    char *doubleStr = new char[nMax];

    sprintf(doubleStr, "%1.6f", val);

    return doubleStr;
}

问题是当输入val1函数返回时1.000000,但我希望输出1. 另外,如果我将代码更改为,sprintf(doubleStr, "%1.0f", val);那么它会正确输出1,但如果输入val更改为0.000005输出0,我希望输出会是0.000005。所以基本上我希望所有输出尽可能短并删除所有不必要0的。这可能sprintf吗?我想用这个功能支持 3.4E +/- 38 范围。

4

2 回答 2

1

由于这是标记为 C++,我假设您可以使用 C 中不可用的 C++ 功能(如果没有,请告诉我,我将删除此答案)。

首先,我建议使用 astd::string而不是char*(这可以让您从管理缓冲区的内存中解放出来)?其次,我建议使用 aostringstream进行转换:

#include <sstream>

std::string ToDecimal(double val) {
    std::ostringstream oss;
    oss << val;
    return oss.str();
}
于 2013-03-24T19:48:59.857 回答
1

事实证明,c++ iostreams(特别是ostringstream)比sprintf.

使用std::fixed操纵器禁用科学计数法。用于std::setprecision指定精度(小数点后的字符数)。在您的情况下,45 位的精度似乎足以代表所有float数字。

#include <sstream>
#include <string>
#include <iostream>
#include <iomanip>

std::string convert(double x)
{
    std::ostringstream buffer;
    buffer << std::fixed << std::setprecision(45) << x;
    std::string result = buffer.str();

    return result;
}

此外,要清理结果,请删除任何尾随零。

    size_t i = result.find_last_not_of('0');
    if (result[i] != '.')
        ++i;
    result.erase(i);

注意:尾随零的清理仅适用于完全可表示的数字(如 0.75 或 0.03125):例如,数字 0.1 转换为0.10000000000000000555111512312578270211815834. 可以使用非恒定精度(取决于数字的大小),但这很难做到正确。

相反,可以使用以下丑陋(且缓慢)的技巧:尝试将字符串的开头转换回double,如果结果等于初始数字,则剪切字符串。

size_t i;
for (i = 1; i < result.size(); ++i)
{
    std::istringstream cut(result.substr(0, i));
    double temp;
    cut >> temp; // the verbose syntax could fit into one line 
    if (temp == x) // by using boost::lexical_cast
        break;
}
于 2013-03-24T21:07:46.303 回答