仅从文本文件

时间:2018-01-19 11:39:20

标签: c++ arrays filestreams

我有一个文本格式的数据文件。它有2个coloums,第一个没用,我可以忽略它,第二个是我需要的。 coloum在每一行上都有不同的值,如string,float等。对于某些计算,我只需要浮动这个coloum的成员。我按照这种方式,首先声明一个float数组并读取值并使用“>>”基本命令保存数组。

问题是当非浮动行出现时,读取功能就像破坏一样。它将非浮点值读为“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 

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

4 个答案:

答案 0 :(得分:2)

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

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。)

答案 1 :(得分:1)

  

没关系,但在此之后将整个值读为“0”,即使它是浮点数。

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

至于问题的解决方案,请将每一行读作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 parseimplement something like is_double(除以并征服)。

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

答案 2 :(得分:1)

我会使用这样的算法:

对于文件中的每一行:

  1. 按空格分割字符串
  2. 对于此拆分的每个结果:
    1. 使用正则表达式“^ \ d +(\。\ d +)?$”
    2. 如果字符串与正则表达式匹配,则转换为float并保存在结果数组

答案 3 :(得分:1)

使用 getline regex_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;
}