如何将整个流读入std :: string?


76

我正在尝试将整个流(多行)读入字符串。

我正在使用此代码,并且可以工作,但是却冒犯了我的风格感……当然,有一种更简单的方法吗?也许使用stringstreams?

void Obj::loadFromStream(std::istream & stream)
{ 
  std::string s;

  std::streampos p = stream.tellg();  // remember where we are

  stream.seekg(0, std::ios_base::end); // go to the end
  std::streamoff sz = stream.tellg() - p;  // work out the size
  stream.seekg(p);        // restore the position

  s.resize(sz);          // resize the string
  stream.read(&s[0], sz);  // and finally, read in the data.


实际上,const对字符串的引用也可以,这可能会使事情变得更容易。

const std::string &s(... a miracle occurs here...)

Answers:


118

怎么样

std::istreambuf_iterator<char> eos;
std::string s(std::istreambuf_iterator<char>(stream), eos);

(如果不是MVP,可以是单线)

2011年后编辑,现在已拼写此方法

std::string s(std::istreambuf_iterator<char>(stream), {});

2
如果您愿意,它可能仍然是单线的string s = string(...)
Mike Seymour,2010年

1
谢谢。您能否详细说明正在做什么?eos不需要以某种方式初始化吗?
罗迪

13
@Roddy:该字符串是从istreambuf_iterator进行范围构造的,该字符串对未格式化的字符进行迭代,直到它等于默认构造的输入迭代器(又称为“流的末尾”)为止。参见Scott Meyers,有效的STL项目29:考虑istreambuf_iterators进行逐字符输入
Cubbi 2010年

1
取自std :: istream_iterator文档 “注:在读取字符时,std :: istream_iterator默认情况下会跳过空格(除非使用std :: noskipws或等效功能禁用),而std :: istreambuf_iterator则不会。效率更高,因为它避免了每个字符一次构造和破坏哨兵对象的开销。”
保罗·索尔特

从性能POV逐字节读取效率不高。有没有更好的解决方案,可以读取更大的块?也许利用上证所优势?
ajeh

27

我参加聚会很晚,但是这里有一个相当有效的解决方案:

std::string gulp(std::istream &in)
{
    std::string ret;
    char buffer[4096];
    while (in.read(buffer, sizeof(buffer)))
        ret.append(buffer, sizeof(buffer));
    ret.append(buffer, in.gcount());
    return ret;
}

我做了一些基准测试,结果发现该std::istreambuf_iterator技术(被接受的答案使用的)实际上要慢得多。在使用的gcc 4.4.5上-O3,我的计算机上相差约4.5倍,并且随着较低的优化设置,差距变得越来越大。


5
确实比我的回答更有效,因为正确的逐块阅读将是。OP希望采用“简易”方式,这通常与“快速”相反。
Cubbi

8
使用string :: reserve(size_t)会使效率更高。
蒂姆(Tim)

Joey,使用-O2进行优化。我记得,选项-O3不是最快的,而是紧凑的代码。
Barney Szabolcs

2
@BarnabasSzabolcs:-Os用于紧凑代码,-O3用于积极的优化,而-O2不那么积极。参见gcc.gnu.org/onlinedocs/gcc/Optimize-Options.html
Joey Adams

在某些情况下,您需要尝试不同的设置。在某些情况下,某些优化会降低速度。(我在O2 / O2评论也有错根据这种说法。)例如参见stackoverflow.com/questions/19470873/...
巴尼索博尔奇

20

你可以做

std::string s;
std::ostringstream os;
os<<stream.rdbuf();
s=os.str();

但我不知道它是否更有效。

替代版本:

std::string s;
std::ostringstream os;
stream>>os.rdbuf();
s=os.str();

1
谢谢。作为解决方案,我发现它非常简单易读,并且正在使用它。但是,我从中吸取了很多教训,因此接受了Cubbi的回答!
罗迪2010年

是的,这是唯一的“读到eofbit”方法。其他方法(istream::get(streambuf*)std::getline(istream, string))仅读取到给定的分隔符为止。
Tanz87 '18 / 12/5

11

您可以尝试使用算法中的内容。我必须为工作做好准备,但是这是事情的快速解决方法(必须有更好的方法):

copy( istreambuf_iterator<char>(stream), istreambuf_iterator<char>(), back_inserter(s) );

0

好吧,如果您正在寻找一种简单且“可读”的方式来做到这一点。我建议在您的项目中添加/使用一些高级框架。为此,我一直在所有项目中使用Poco和Boost。在这种情况下,使用Poco:

    string text;
    FileStream fstream(TEXT_FILE_PATH);
    StreamCopier::copyToString(fstream, text);

0

将getline与分隔符一起使用会怎样?下一个代码可帮助我使用g ++-10将整个std :: cin读入ubuntu上的字符串中。

#include <iostream>
#include <string>

using namespace std;

int main() {
    string s;

    getline(cin, s, {}); //the whole stream into variable s

    return 0;
}

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.