仅从文本文件中读取浮点值

Reading only float values from text file

我有一个文本格式的数据文件。它有 2 个颜色,第一个没有用,我可以忽略它,第二个是我需要的。 coloum 在每一行上都有不同的值,如字符串、浮点数等。对于某些计算,我只需要浮动这个 coloum 的成员。我遵循这种方式,首先声明一个浮点数组并读取值并使用 ">>" 基本命令保存数组。

问题是当出现非浮动行时,读取功能无法正常工作。它将非浮点值读取为“0”并保存为该值。没关系,但在那之后即使它是一个浮点数,也会将整个值读取为“0”。

Datafile.txt(示例)

aa 1.1
bb 2.2
cc 3.0
dd somestring
ee 4.3
ff 4.9

代码(示例)

 do 
{
    dfile >> a >> dat[i]; 
    ofile << dat[i]<<endl;
    cout << dat[i]<<endl;
    i++;

}while(dfile.eof());

输出文件(示例)

1.1
2.2
3.0
0
0
0
..goes 

我想到了两种解决问题的方法。第一个是跳过非浮动行。第二个是一段时间内的阅读行。因为浮点值按顺序列出。

首先读取字符串,然后尝试转换为数字:

std::string maybeNumber;
while (dfile >> a >> maybeNumber)
{

    std::istringstream is(maybeNumber);
    float number = 0.0f;
    if (is >> number)
    {
        dat[i] = number;
        i++;
    }    
}

(你不想使用eof。每个人都认为他们想要使用eof,但它几乎总是不是他们需要的。有关详细信息,请参阅this question。)

It's OK but after that reads whole values as "0" even if it was a float.

您没有显示完整的代码,但这听起来像是输入流在第一个不可解析的浮点数之后处于错误状态,因此它根本停止读取任何内容。

至于问题的解决方案,将每一行读取为 std::string,然后将该行拆分为 std::vector<std::string>。如果向量不包含两个元素或者如果第二个元素不是 double,什么也不做,否则处理该行。

像这样:

std::string line;
while (std::getline(dfile, line))
{
    auto const elements = parse(line);
    if (size(elements) == 2 && is_double(elements[1]))
    {
        // process
    }
}

既然你已经在更高的抽象层次上解决了问题,你所要做的就是implement something like parse and implement something like is_double(分而治之)。

另请注意,默认的 C++ 浮点类型是 double 而不是 float。如有疑问,请使用 double.

我会使用这样的算法:

对于文件中的每一行:

  1. 用空格分割字符串
  2. 对于这次拆分的每个结果:
    1. Macht 与正则表达式 "^\d+(\.\d+)?$"
    2. 如果字符串匹配正则表达式,转换为浮点数并保存在结果数组中

使用 getlineregex_token_iterator

#include <regex>
#include <fstream>
#include <string>
#include <vector>
#include <algorithm>

int main()
{
    std::vector<float> v;
    std::regex reg("-?\d+(\.\d+)?");
    const char* filename="myfile.txt";

    std::ifstream ifs(filename,std::ios::binary);

    for(std::string s;std::getline(ifs,s);)
       for(std::sregex_token_iterator rgi(s.begin(),s.end(),reg);rgi!=std::sregex_token_iterator();++rgi)
         v.push_back(std::stof(rgi->str());

    std::copy(v.begin(),v.end(),std::ostream_iterator<float>(std::cout," ");
    return 0;
}