为什么我不能做一个引用向量?

当我这样做的时候:

std::vector<int> hello;

一切都很好。然而,当我把它变成一个引用向量时:

std::vector<int &> hello;

我就会犯可怕的错误

指向引用的指针是非法的

我想把一堆对struct的引用放到一个vector中,这样我就不用管指针了。为什么矢量会为此大发脾气?我唯一的选择是使用指针的向量代替?

244231 次浏览

这是c++语言的一个缺陷。您不能获取引用的地址,因为尝试这样做将导致被引用对象的地址,因此您永远无法获得指向引用的指针。std::vector使用指向其元素的指针,因此存储的值需要能够被指向。你将不得不使用指针代替。

根据它们的本质,引用只能在创建时设置;也就是说,下面两行有非常不同的效果:

int & A = B;   // makes A an alias for B
A = C;         // assigns value of C to B.

此外,这是非法的:

int & D;       // must be set to a int variable.

然而,当你创建一个向量时,没有办法在创建时给它的项赋值。实际上你只是做了一大堆上一个例子。

vector容器的组件类型必须是< em >可转让的< / em >。引用是不可赋值的(只能在声明时初始化它们一次,以后不能让它们引用其他东西)。其他不可赋值类型也不允许作为容器的组件,例如vector<const int>是不允许的。

boost::ptr_vector<int>将工作。

编辑:建议使用std::vector< boost::ref<int> >,这将不起作用,因为你不能默认构造一个boost::ref

正如其他人所提到的,您可能最终会使用指针向量。

然而,你可能想要考虑使用ptr_vector代替!

是的,你可以,寻找std::reference_wrapper,它模仿引用,但是可赋值的,也可以“重新定位”

Ion Todirel已经提到了一个答案是的使用std::reference_wrapper。在由于c++ 11中,我们有一个从std::vector中检索对象并使用std::remove_reference删除引用的机制。下面给出了一个使用g++clang选项-std=c++11编译并成功执行的示例。

    #include <iostream>
#include <vector>
#include <functional>
    

class MyClass {
public:
void func() {
std::cout << "I am func \n";
}


MyClass(int y) : x(y) {}


int getval() {
return x;
}


private:
int x;
};


int main() {
std::vector<std::reference_wrapper<MyClass>> vec;


MyClass obj1(2);
MyClass obj2(3);


MyClass& obj_ref1 = std::ref(obj1);
MyClass& obj_ref2 = obj2;


vec.push_back(obj_ref1);
vec.push_back(obj_ref2);


for (auto obj3 : vec) {
std::remove_reference<MyClass&>::type(obj3).func();
std::cout << std::remove_reference<MyClass&>::type(obj3).getval() << "\n";
}
}

正如其他注释所建议的那样,您只能使用指针。 但如果它有帮助,这里有一个技巧来避免直接面对指针

你可以这样做:

vector<int*> iarray;
int default_item = 0; // for handling out-of-range exception


int& get_item_as_ref(unsigned int idx) {
// handling out-of-range exception
if(idx >= iarray.size())
return default_item;
return reinterpret_cast<int&>(*iarray[idx]);
}

TL;博士

像这样使用std::reference_wrapper:

#include <functional>
#include <string>
#include <vector>
#include <iostream>


int main()
{
std::string hello = "Hello, ";
std::string world = "everyone!";
typedef std::vector<std::reference_wrapper<std::string>> vec_t;
vec_t vec = {hello, world};
vec[1].get() = "world!";
std::cout << hello << world << std::endl;
return 0;
}

Demo

长回答

标准建议一样,对于包含类型为T的对象的标准容器XT必须是X中的Erasable

Erasable表示以下表达式是格式良好的:

allocator_traits<A>::destroy(m, p)

A是容器的分配器类型,m是分配器实例,p是类型为*T的指针。有关Erasable定义,请参阅在这里

默认情况下,std::allocator<T>被用作vector的分配器。对于默认的分配器,这个需求等同于p->~T()的有效性(注意T是一个引用类型,而p是指向引用的指针)。然而,指向引用的指针是非法的,因此表达式不是良好的形式。

我在c++ 14中找到了原因,“国际标准ISO/IEC 14882:2014(E)编程语言c++ "

< p > [8.3.2-5.s1] 没有引用的引用,没有引用的数组,没有指向引用的指针