如何解决如何从 std::vector<float> 转换为 std::istream?
我有一个 std::vector 并且该函数需要一个 std::istream:
callMe(std::istream& is)
进行转换的最佳方法是什么?还有比这更聪明的吗?
std::stringstream sstr;
for(int i = 0; i < myVector.size(); ++i) {
sstr << myVector[i] << " ";
}
std::istringstream istr{sstr.str()};
callMe(istr);
编辑:感谢到目前为止的建议! 更新代码:
std::stringstream sstr;
for(const float& val : myVector) {
sstr << val << " ";
}
callMe(sstr);
解决方法
问题在于 std::istream
本质上是基于字符的。如果您想继续使用 callMe(std::istream& is)
作为接口,您必须将 myVector
的每个元素转换为字符,然后在某个时候返回。如果您想坚持使用此选项,我个人认为 ostream_iterator
是一个优雅的解决方案:
copy(begin(data),end(data),std::ostream_iterator<float>(sstr));
完整示例:
void callMeStream(std::istream &is)
{
float f1;
is >> f1;
std::cout << "Stream: " << f1 << std::endl;
}
// ...
std::vector<float> data = {3.5,1.5,2.5 /* ... */};
std::stringstream sstr;
copy(begin(data),std::ostream_iterator<float>(sstr));
callMeStream(sstr); // Output: "Stream: 3.51"
如果您愿意更改callMe
的签名,则可以避免这种转换:
template <class T>
void callMeTemplate(T &is)
{
float f1;
is >> f1;
std::cout << "Template: " << f1 << std::endl;
}
#define NO_ELEMENT -1.0;
class data_wrapper
{
std::vector<float>::const_iterator current;
const std::vector<float>::const_iterator last;
public:
data_wrapper(const std::vector<float> &data) : current(begin(data)),last(end(data)) {}
data_wrapper &operator>>(float &value)
{
if (last == current)
{
value = NO_ELEMENT;
}
else
{
value = *current++;
}
return *this;
}
};
// ...
data_wrapper wrapper(data);
callMeTemplate(wrapper); // Output: "Template: 3.5"
在第二个示例中,float
值永远不会转换为字符序列,并且可以接受 data_wrapper
和 std::istream
类型。当然,如果您愿意完全更改 callMe
的签名,您不妨将其更改为直接接受开始/结束迭代器范围或 vector
。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。