让`std :: vector <unsigned char =“”>`从`std :: string`窃取内存

时间:2018-06-23 20:09:53

标签: c++ stdvector move-semantics stringstream rvalue-reference

假设我们有std::string s保留了一个原始数据缓冲区,但是我们想使用std::vector<uint8_t> v。缓冲区长度以百万计。有没有一种简单的方法可以让v窃取s的内存,从而避免复制缓冲区?

就像std::vector<uint8_t>::vector(std::string&&)会做的一样,但是以某种方式从STL的外部进行。

或者,是否可以通过与v一样高效的操作从std::stringstream ss获取ss.str()

1 个答案:

答案 0 :(得分:0)

好的,在那里有很多评论,让我们尝试将它们放在一起,因为我需要练习,并且可能会获得一些积分[更新:没:(]。

我对“现代C ++”还不陌生,因此请随便使用。可能需要迟到C ++ 17,我还没有仔细检查过。任何批评都值得欢迎,但我希望编辑自己的帖子。并且在阅读本文时请记住,OP 实际上要做的是从文件中读取其字节。谢谢。

更新:已进行了调整,以处理文件大小根据以下@Deduplicator的注释在调用stat()和调用fread()之间变化的情况...然后将fread替换为std::ifstream,我想我们到了。

#include <string>
#include <vector>
#include <optional>
#include <iostream>
#include <fstream>

#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <errno.h>

using optional_vector_of_char = std::optional <std::vector <char>>;

// Read file into a std::optional <std::vector <char>>.
// Now handles the file size changing when we're not looking.
optional_vector_of_char SmarterReadFileIntoVector (std::string filename)
{
    for ( ; ; )
    {
        struct stat stat_buf;
        int err = stat (filename.c_str (), &stat_buf);
        if (err)
        {
            // handle the error
            return optional_vector_of_char ();   // or maybe throw an exception
        }

        size_t filesize = stat_buf.st_size;

        std::ifstream fs;
        fs.open (filename, std::ios_base::in | std::ios_base::binary);
        if (!fs.is_open ())
        {
            // handle the error
            return optional_vector_of_char ();
        }

        optional_vector_of_char v (filesize + 1);
        std::vector <char>& vecref = v.value ();
        fs.read (vecref.data (), filesize + 1);

        if (fs.rdstate () & std::ifstream::failbit)
        {
            // handle the error
            return optional_vector_of_char ();
        }

        size_t bytes_read = fs.gcount ();
        if (bytes_read <= filesize)              // file same size or shrunk, this code handles both
        {
            vecref.resize (bytes_read);
            vecref.shrink_to_fit ();
            return v;                            // RVO
        }

        // File has grown, go round again
    }
}    

int main ()
{
    optional_vector_of_char v = SmarterReadFileIntoVector ("abcde");
    std::cout << std::boolalpha << v.has_value () << std::endl;
}

Live demo。当然,没有可用的实际文件可供读取,所以...


:您是否考虑过编写自己的简单容器来映射文件视图?只是一个想法。