20

我正在尝试使用istringstream将一个简单的字符串拆分为一系列整数:

#include <string>
#include <iostream>
#include <sstream>
#include <vector>

using namespace std;

int main(){

    string s = "1 2 3"; 
    istringstream iss(s);   

    while (iss)
    {
        int n;
        iss >> n;
        cout << "* " << n << endl;
    } 
}

我得到:

* 1
* 2
* 3
* 3

为什么最后一个元素总是出现两次?如何解决?

4

2 回答 2

34

它出现了两次,因为你的循环是错误的,正如在http://www.parashift.com/c++-faq-lite/input-output.html#faq-15.5(间接地)解释的那样(在这种情况下while (iss)没有不同) while (iss.eof()).

具体来说,在第三次循环迭代中,iss >> n成功并获得您的3,并使流处于良好状态。由于这个良好的状态,循环然后第四次运行,直到下一个(第四次)iss >> n随后失败,循环条件才被打破。但是在第四次迭代结束之前,你仍然输出n......第四次。

尝试:

#include <string>
#include <iostream>
#include <sstream>
#include <vector>

using namespace std;

int main()
{
    string s = "1 2 3"; 
    istringstream iss(s);   
    int n;

    while (iss >> n) {
        cout << "* " << n << endl;
    } 
}
于 2011-03-02T14:19:25.843 回答
1

希望这会有所帮助:
iss : 1 2 3
Iteration 1
iss : 1 2 3 (Initially)
n=1
iss : 2 3
//* 1 被打印
Iteration 2:
iss : 2 3 (Initially)
n=2
iss : 3
// * 2 被打印
Iteration 3
iss : 3
n=3
iss : ''
Iteration 4
iss : '' n not changed//为 iss 的eof
设置的标志,因为没有来自流 iss 的进一步输入:''

正如上面的帖子正确提到的那样,while (iss) 与 while (iss.eof()) 没有什么不同。
在内部,函数(istream::operator>>) 通过首先构造一个哨兵对象来访问输入序列(noskipws 设置为 false[这意味着空格是分隔符,您的列表将是 1,2,3])。然后(如果[这里 eof 未达到]),它调用num_get::get [获取下一个整数] 来执行提取和解析操作,相应地调整流的内部状态标志。最后,它在返回之前销毁哨兵对象。

参考:http ://www.cplusplus.com/reference/istream/istream/operator%3E%3E/

于 2013-11-04T11:11:54.280 回答