Search code examples
c++shared-ptrweak-ptr

Should I call reset on a weak_ptr if I happen to notice it's expired?


I have a collection of Creature objects that are created and owned in one part of my application using std::make_shared and std::shared_ptr.

I also keep track of a selection of zero or one Creature in a World object using a std::weak_ptr<Creature>.

void World::SetSelection(const std::shared_ptr<Creature>& creature) {
    selection = creature;
}

std::shared_ptr<Creature> World::GetSelection() const {
    return selection.lock();
}

The caller of GetSelection is responsible for checking if the pointer is empty. If it is, that means there is currently no selection.

This all works perfectly to my liking: when the selected Creature dies of natural causes (elsewhere in the application), GetSelection starts returning a nullptr again as if nothing was ever selected.

However in that case the World::selection member still points to the std::shared_ptr's control block. This can be quite big because I use std::make_shared to create my Creature objects (I realize that the Creature object was correctly destroyed at the right time but the memory for it is still allocated). I'm considering changing GetSelection to this:

std::shared_ptr<Creature> World::GetSelection() {
    const auto ret = selection.lock();
    if (!ret)
        selection.reset();

    return ret;
}

This frees up the memory as soon as I notice it's not needed anymore. Annoyingly, this version of GetSelection cannot be const.

My questions:

  1. Which version of GetSelection would be considered best practice in this situation?

  2. Does the answer change if something similar happens in templated code, where sizeof(T) is unknown and could be huge? Or in C++14 where std::make_shared<T[]> could be involved?

  3. If the second version is always best, what is the rationale for std::weak_ptr<T>::expired and lock not doing it themselves?


Solution

  • It should be noted, first off, that the emplacement strategy of std::make_shared is optional, i.e. the standard does not mandate that implementations perform this optimization. It's a non-binding requirement, which means that perfectly conforming implementations may elect to forego it.

    To answer your questions:

    1. Given that you seem to have only one selection (and you're not therefore bloating your memory use by keeping many of these control blocks around), I'd argue for keeping it simple. Is memory a bottleneck? This screams micro-optimization to me. You should write the simpler code, where you can apply const, and then go back and optimize later if the need arises.

    2. The answer doesn't unconditionally change, it changes conditional upon the problem domain and what your bottleneck is. If you're allocating one object that's "huge" (say a hundred kilobytes), and the space for that object is kicking around in a relatively unused control block until replaced, that probably isn't your bottleneck, and probably isn't worth writing more code (which is intrinsically more error prone, difficult to maintain, and decipher) to "solve".

    3. As std::weak_ptr::lock and std::weak_ptr::expired are const, under the interpretation of const for C++11, they must be thread safe. Therefore, given some std::weak_ptr, it must be safe to call any combination of lock() and expired() concurrently. Under the hood, the std::weak_ptr stores a pointer to the control block, which it looks through to examine/increment/etc. atomic counters to determine if the object has expired, or to see if it can acquire the lock. If you wanted to implement your optimization internal to std::weak_ptr, you'd have to somehow inspect the control block's state, and then atomically remove the pointer to the control block if the pointer has expired. This would cause overhead (even if this could be done simply with atomics, it would still have overhead) on every access to the std::weak_ptr, all for the sake of a small optimization.