我对以下用于对字符串进行标记的代码有疑问(用空格分隔标记)。
#include <iostream>
#include <iterator>
#include <sstream>
#include <string>
#include <vector>
using namespace std;
int main()
{
string s="And I feel fine ...";
istringstream ss(s);
vector<string> tokens{istream_iterator<string>(ss),{}};
for(auto& elem: tokens)
cout << elem << endl;
}
这完全没问题。另一方面,如果我尝试将临时istringstream
对象传递给
istream_iterator
(main
内的第3行),例如
vector<string> tokens{istream_iterator<string>(istringstream(s)),{}};
我收到编译时错误
error: no matching conversion for functional-style cast from 'istringstream' (aka 'basic_istringstream<char>') to 'istream_iterator<string>'
我相信这是因为我不能将临时rvalue绑定到非const左值引用,istream_iterator
的构造函数将引用作为参数。有没有办法从临时构建istream_iterator<string>
?我不能在临时对象上使用std::ref
......
谢谢!
答案 0 :(得分:3)
使用std::skipw
技巧:
std::vector<std::string> tokens{
std::istream_iterator<std::string>(std::istringstream(s) >> std::skipws), {}
};
这是有效的,因为提取器返回对流的左值引用,并且设置std::skipw
不会产生任何影响,因为它在所有流上默认启用。要在一般情况下使用它,您可能希望拥有一个采用通用引用的函数,然后返回对该对象的引用:
template<typename T>
T& lvalue(T&& x)
{
return x;
}
// ...
std::istream_iterator<std::string>(lvalue(std::istringstream(s)));
确保参考文献没有在完整表达式之后使用,否则您将有一个悬空参考,因为临时文件已被破坏。