1

正如标题所描述的,我试图将指向 a 数据的指针传递给std::vector一个需要双指针的函数。以下面的代码为例。我有一个 int 指针d,它被传递给myfunc1as &d(仍然不确定是否将其称为指针的引用或什么),其中函数将其引用更改为填充1,2,3,4. 但是,如果我有一个std::vectorints 并尝试传递&(vec.data())myfunc1编译器会抛出错误lvalue required as unary ‘&’ operand。我已经(int *)&(vec.data())按照这个答案尝试过类似的方法,但它不起作用。

仅供参考,我知道我可以做一些事情,比如myfunc2直接将向量作为参考传递,工作就完成了。但我想知道是否可以使用myfunc1std::vector 的指针。

任何帮助将不胜感激。

#include <iostream>
#include <vector>


using std::cout;
using std::endl;
using std::vector;

void myfunc1(int** ptr)
{
    int* values = new int[4];
    // Fill all the with data
    for(auto& i:{0,1,2,3})
    {
        values[i] = i+1;
    }

    *ptr = values;
}

void myfunc2(vector<int> &vec)
{
    int* values = new int[4];
    // Fill all the with data
    for(auto& i:{0,1,2,3})
    {
        values[i] = i+1;
    }

    vec.assign(values,values+4);
    delete values;
}

int main()
{
    // Create int pointer
    int* d;

    // This works. Reference of d pointing to the array
    myfunc1(&d);

    // Print values
    for(auto& i:{0,1,2,3})
    {
        cout << d[i] << " ";
    }
    cout << endl;

    // Creates the vector
    vector<int> vec;

    // This works. Data pointer of std::vector pointing to the array
    myfunc2(vec);

    // Print values
    for (const auto &element : vec) cout << element << " ";
    cout << endl;

    // This does not work
    vector<int> vec2;
    vec2.resize(4);

    myfunc1(&(vec2.data()));

    // Print values
    for (const auto &element : vec2) cout << element << " ";
    cout << endl;

    return 0;
}

编辑:我的实际代码所做的是从磁盘读取一些二进制文件,并将部分缓冲区加载到向量中。我在从读取函数中获取修改后的向量时遇到了麻烦,这就是我想出的让我解决它的方法。

4

2 回答 2

2

当你写: myfunc1(&(vec2.data()));

您正在获取右值的地址。尖头int*是一个临时的,在调用后立即被销毁。

这就是您收到此错误的原因。

但是,正如@molbdnilo所说,在您的myfunc1()函数中,您正在重新分配指针(不关心顺便破坏先前分配的内存)。
但它std::vector已经自行管理其数据存储器。你不能也不能把手放在上面。


我的实际代码所做的是从磁盘读取一些二进制文件,并将部分缓冲区加载到向量中。

一种解决方案可能是std::vector通过将迭代器传递到开头并将迭代器传递到所需部分的结尾来构造您的构造函数,以提取构造函数的参数。

例如:

int * buffer = readAll("path/to/my/file"); // Let's assume the readAll() function exists for this example

// If you want to extract from element 5 to element 9 of the buffer
std::vector<int> vec(buffer+5, buffer+9);

如果std::vector已经存在,您可以assign()像在中一样使用成员函数myfunc2()

vec.assign(buffer+5, buffer+9);

当然,在这两种情况下,您都必须确保在访问缓冲区时没有尝试访问越界元素。

于 2019-12-02T09:27:25.257 回答
0

问题是您不能获取 的地址data(),因为它只是指针的临时副本,因此写入指向它的指针没有多大意义。这很好。你不想传递data()给这个函数,因为它会用一个新数组覆盖指针,这会破坏向量。您可以*从函数中删除一个并只分配给它而不在那里分配内存。这将起作用,但请确保在调用者中分配内存(使用resize, 只会reserve导致未定义的行为,因为data()它只是指向有效范围开头的指针[data(), data() + size())。范围[data(), data() + capacity ())不一定有效。

于 2019-12-02T09:23:15.320 回答