我有一个文本格式的数据文件。它有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
我想过两种解决问题的方法。第一个是跳过非浮动行。第二个是在一段时间内阅读行。因为浮点值列在序列中。
答案 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 parse
和implement something like is_double
(除以并征服)。
另请注意,默认的C ++浮点类型为double
而不是float
。如有疑问,请使用double
。
答案 2 :(得分:1)
我会使用这样的算法:
对于文件中的每一行:
答案 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;
}