Removing objects from C++ containers without deleting them
Asked Answered
A

3

10

I am using C++ std vector to store render objects for a simple scene graph implementation.I need an ability to add and remove render objects in run time from the scene graph.Adding is not a problem,for removing: reading docs for vector and other C++ containers it appears that when objects are popped their destructors are called.That is not what I need because I want to be able to re-add those objects later to the rendering loop.What are the possible solution to this problem? Important detail I forgot to mention -I am using vector of pointers to the render objects.

Ataliah answered 25/11, 2012 at 10:1 Comment(7)
Store smart pointers to objects managed from somewhere else.Conjugation
Wouldn't copying them before you delete them be a solution? Or, in case of C++11, moving them out?Poltergeist
@Conjugation I'd say either "smart pointers" or "managed from somewhere else", can't see how these two could mix.Spongioblast
Using smart pointers prevents the object destructors from being called?Ataliah
@MichaelIV with the appropriate smart pointer, the last-man-out will release the object and fire the destructor. (std::shared_ptr is probably what you're looking for).Ultann
Add some example code. Once an object has been removed from a container, the object no longer exists, so wanting to re-add it later makes no sense. The only thing that can exist are copies of that object, those copies will not be destroyed, and can themselves be copied into a new container at a later stage. You seem to have a non-problem here.Varix
possible duplicate of Does std::list::remove method call destructor of each removed element?Suited
B
11

It seems you're confused with the basic concept of object instances. When you add something to a vector, you don't move it into it, you copy it:

vector<string> vec;
string s;
vec.push_back(s);

vec[0] is not s, it's a copy of s. So when you remove it from the vector, s is not affected.

If you don't want copies, you should switch to pointers instead. You can them remove the pointers from the vector and the destructor of the object they point to will not be called.

Edit: OK, it seems you're already using pointers. You said:

reading docs for vector and other C++ containers it appears that when objects are popped their destructors are called

That is true. When you remove a pointer from the vector, the pointer is getting destroyed. That's what the docs mean. It doesn't mean that the object the pointer points to is getting destroyed:

vector<string*> vec;
string s;
vec.push_back(&s);
vec.pop_back();

s is not affected at all. What gets destroyed by the pop operation is the pointer that holds the address of s, not s itself.

So you're fine.

Burch answered 25/11, 2012 at 10:19 Comment(5)
Yes I know it :) I forgot to mention I am using vector of pointers.Ataliah
@MichaelIV: The pointers will be destroyed, but the things that they point to will not be. (So you are fine as long as you properly manage the lifetimes of the objects being pointed to).Varix
@MichaelIV Then you don't have a problem; when you remove the pointer from the vector, only the pointer is destroyed, not the object the pointer points to.Burch
@MichaelIV Concur with Nikos. If your vectors are holding pointers already, you likely don't need smart pointers assuming your memory is managed properly wherever the objects really are. But you should also not be seeing the destructions your seeing in that sense either.Ultann
@MichaelIV I updated the answer to explain what the docs mean when they say that popped objects are destroyed.Burch
K
3

You need to be aware of ownership to make this work correctly. If the vector you are using is only temporary and only to observe objects, just use a vector of points.

std::vector<object*> x;

On destruction of that vector the objects that are being pointed to are unaffected.

If you want to share ownership, use a boost::shared_ptr or std::shared_ptr.

Kiaochow answered 25/11, 2012 at 10:21 Comment(0)
F
1

When a pointer is popped, no destructor is called. Notionally even primitive types have destructors, to explain what happens when they go out of scope. Destroying a vector of pointers is the same as having a bunch of local pointer variables go out of scope.

Reference-counted smart pointers are objects that overload the * and -> operators to behave like pointers. They do implement the destructor to destroy the pointed-to object, thus implementing ownership. But for a scene graph, that's probably unnecessary.

Farrah answered 25/11, 2012 at 10:24 Comment(1)
The documentation of std:: containers pop() explicitly states "This calls the removed element's destructor.". That seems to contradict your statement.Spandex

© 2022 - 2024 — McMap. All rights reserved.