草庐IT

c++ - 在 C++ 中读取大字符串——有安全快速的方法吗?

coder 2024-02-05 原文

http://insanecoding.blogspot.co.uk/2011/11/how-to-read-in-file-in-c.html回顾了在 C++ 中将整个文件读入字符串的多种方法。最快选项的关键代码如下所示:

std::string contents;
in.seekg(0, std::ios::end);
contents.resize(in.tellg());
in.seekg(0, std::ios::beg);
in.read(&contents[0], contents.size());

不幸的是,这并不安全,因为它依赖于 string以特定方式实现。例如,如果实现共享字符串,则修改 &contents[0] 处的数据。可能会影响正在读取的字符串以外的字符串。 (更一般地说,不能保证这不会破坏任意内存——这在实践中不太可能发生,但依赖它不是好的做法。)

C++ 和 STL 旨在提供与 C 一样高效的功能,因此人们期望有一个与 C 一样快但保证安全的版本。

vector<T>的情况下,有一些函数可以用来访问原始数据,可以用来有效地读取一个 vector :

T* vector::data();
const T* vector::data() const; 

其中第一个可用于读取 vector<T>有效率的。不幸的是,string等同于只有提供const变体:

const char* string::data() const noexcept;

所以这不能用来有效地读取字符串。 (大概省略了 non-const 变体以支持共享字符串实现。)

我还检查了字符串构造函数,但是那些接受 char* 的构造函数复制数据——没有移动它的选项。

有没有一种安全快速的方法可以将文件的全部内容读入字符串?

可能值得注意的是,我想阅读一个 string而不是 vector<char>这样我就可以使用 istringstream 访问结果数据. vector<char> 没有等效项.

最佳答案

如果你真的想避免复制,你可以将文件拖入 std::vector<char> 中。 , 然后滚动你自己的 std::basic_stringbuf 从 vector 中提取数据。

然后您可以声明一个 std::istringstream并使用 std::basic_ios::rdbuf 用您自己的缓冲区替换输入缓冲区。

需要注意的是,如果您选择调用 istringstream::str 它将调用 std::basic_stringbuf::str 并且需要一份拷贝。但是,听起来您不需要该功能,实际上可以将其 stub 。

是否通过这种方式获得更好的性能需要实际测量。但至少你避免了在复制过程中必须有两个大的连续内存块。此外,您可以使用类似 std::deque 的内容如果您想处理无法在连续内存中分配的真正巨大的文件,则作为您的底层结构。

还值得一提的是,如果您真的只是流式传输该数据,您实际上是通过先将其读入字符串来进行双缓冲。除非您出于其他目的还需要内存中的内容,否则 std::ifstream 中的缓冲可能就足够了。如果您确实吞咽了该文件,您可以通过关闭缓冲来获得提升。

关于c++ - 在 C++ 中读取大字符串——有安全快速的方法吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39379811/

有关c++ - 在 C++ 中读取大字符串——有安全快速的方法吗?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  4. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

    类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

  5. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

    我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

  6. ruby-on-rails - unicode 字符串的长度 - 2

    在我的Rails(2.3,Ruby1.8.7)应用程序中,我需要将字符串截断到一定长度。该字符串是unicode,在控制台中运行测试时,例如'א'.length,我意识到返回了双倍长度。我想要一个与编码无关的长度,以便对unicode字符串或latin1编码字符串进行相同的截断。我已经了解了Ruby的大部分unicode资料,但仍然有些一头雾水。应该如何解决这个问题? 最佳答案 Rails有一个返回多字节字符的mb_chars方法。试试unicode_string.mb_chars.slice(0,50)

  7. ruby - Facter::Util::Uptime:Module 的未定义方法 get_uptime (NoMethodError) - 2

    我正在尝试设置一个puppet节点,但ruby​​gems似乎不正常。如果我通过它自己的二进制文件(/usr/lib/ruby/gems/1.8/gems/facter-1.5.8/bin/facter)在cli上运行facter,它工作正常,但如果我通过由ruby​​gems(/usr/bin/facter)安装的二进制文件,它抛出:/usr/lib/ruby/1.8/facter/uptime.rb:11:undefinedmethod`get_uptime'forFacter::Util::Uptime:Module(NoMethodError)from/usr/lib/ruby

  8. ruby - 将差异补丁应用于字符串/文件 - 2

    对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

  9. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  10. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

    我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

随机推荐