There is a thread in the comments section in this post about using std::vector::reserve()
vs. std::vector::resize()
.
Here is the original code:
void MyClass::my_method()
{
my_member.reserve(n_dim);
for(int k = 0 ; k < n_dim ; k++ )
my_member[k] = k ;
}
I believe that to write elements in the vector
, the correct thing to do is to call std::vector::resize()
, not std::vector::reserve()
.
In fact, the following test code "crashes" in debug builds in VS2010 SP1:
#include <vector>
using namespace std;
int main()
{
vector<int> v;
v.reserve(10);
v[5] = 2;
return 0;
}
Am I right, or am I wrong? And is VS2010 SP1 right, or is it wrong?
There are two different methods for a reason:
std::vector::reserve
will allocate the memory but will not resize your vector, which will have a logical size the same as it was before.std::vector::resize
will actually modify the size of your vector and will fill any space with objects in their default state. If they are ints, they will all be zero.After reserve, in your case, you will need a lot of push_backs to write to element 5. If you don't wish to do that then in your case you should use resize.
There probably should be a discussion about when both methods are called with a number that's LESS than the current size of the vector.
Calling
reserve()
with a number smaller than the capacity will not affect the size or the capacity.Calling
resize()
with a number smaller than current size the container will be reduced to that size effectively destroying the excess elements.To sum up
resize()
will free up memory whereasreserve()
will not.Yes you’re correct, Luchian just made a typo and is probably too coffee-deprived to realise his mistake.
resize actually changes the amount of elements in the vector, new items are default constructed if the resize causes the vector to grow.
in this case size is 10.
reserve on the other hand only requests that the internal buffer be grown to the specified size but does not change the "size" of the array, only its buffer size is changed.
in this case size is still 0.
So to answer your question, yes you are right, even if you reserve enough space you are still accessing uninitialized memory with the index operator. With an int thats not so bad but in the case of a vector of classes you would be accessing objects which have not been constructed.
Bounds checking of compilers set to debug mode can obviously be confused by this behavior which may be why you are experiencing the crash.
It depends on what you want to do.
reserve
does not add any elements to thevector
; it only changes thecapacity()
, which guarantees that adding elements will not reallocate (and e.g. invalidate iterators).resize
adds elements immediately. If you want to add elements later (insert()
,push_back()
), usereserve
. If you want to access elements later (using[]
orat()
), useresize
. So youreMyClass::my_method
can be either:or
Which one you chose is a question of taste, but the code you quote is clearly incorrect.
Answered here by Jan Hudec : Choice between vector::resize() and vector::reserve()