|
Boost Users : |
From: ypotin (yannp63_at_[hidden])
Date: 2020-05-06 00:37:57
Hello,
I was running some tests involving std::weak_ptr using boost
serialization, and I realized that when I read a std::weak_ptr from an
archive that doesn't contain the "parent" std::share_ptr, the behavior
of the read pointer is "hard to understand".
Here is an example :
#include <cassert>
#include <sstream>
#include <memory>
#include <boost/archive/binary_iarchive.hpp>
#include <boost/archive/binary_oarchive.hpp>
#include <boost/serialization/export.hpp>
#include <boost/serialization/shared_ptr.hpp>
#include <boost/serialization/weak_ptr.hpp>
class MyInt
{
public:
   MyInt()
       : value(0)
   {
   }
   int value;
   template<class Archive>
   void serialize(Archive &ar, unsigned int)
   {
       ar & value;
   }
};
int main()
{
   std::stringstream data;
   {
       std::shared_ptr<MyInt> shared = std::make_shared<MyInt>();
       assert(shared.use_count() == 1);
       shared->value = 12;
       /* Write a weak_ptr */
       {
           std::weak_ptr<MyInt> weak(shared);
           assert(shared.use_count() == 1);
           boost::archive::binary_oarchive ar(data);
           ar << weak;
           assert(shared.use_count() == 1);
       }
       assert(shared.use_count() == 1);
       /* Read a weak_ptr */
       std::weak_ptr<MyInt> weak;
       {
           boost::archive::binary_iarchive ar(data);
           ar >> weak;
           assert(shared.use_count() == 1);
           assert(weak.use_count() == 1);
           {
               std::shared_ptr<MyInt> loaded = weak.lock();
               assert(loaded);
               /* It looks good */
               assert(loaded->value == 12);
               /* Let's see... */
               assert(loaded != shared);
               shared->value = 42;
               assert(loaded->value == 12);
               /* So another shared_ptr exists somewhere... */
               assert(shared.use_count() == 1);
               assert(loaded.use_count() == 2);
               assert(weak.use_count() == 2);
           }
           /* But where ? */
           assert(shared.use_count() == 1);
           assert(weak.use_count() == 1);
       }
       /* Let's see */
       assert(shared.use_count() == 1);
       assert(weak.use_count() == 0 /* weak.expired() */);
       /* Hmm, was it inside the archive itself ? */
   }
   /* Let's try again, we need the input stream to be back to the
beginning */
   data.seekg(0);
   std::weak_ptr<MyInt> weak;
   {
       boost::archive::binary_iarchive ar(data);
       assert(weak.use_count() == 0);
       ar >> weak;
       assert(weak.use_count() == 1);
       /* It looks fine */
   }
   assert(weak.use_count() == 0);
   /* aw, on a second thought, it really doesn't */
   return 0;
}
I am building and running this using g++-9.3.0 with boost-1.72.0-1.
Clearly, the weak_ptr gets itself a shared_ptr "somewhere" as it can be
used after being read. I was wondering "where it is stored and how to
access it, if possible, without some `weak.lock()` call ?"
It looks like the shared_ptr gets destroyed at the same time as the
archive, so I guess it is cleaned up thanks to the
delete_created_pointers method when the iarchive is destroyed : does
delete_created_pointers actually contains also shared_ptr ? As far as I
can see in the code serialization/weak_ptr.hpp, we actually save then
load a std::shared_ptr, so I was expecting the weak_ptr to expire as
soon as the call returns : since it is not, there *is* another reference.
Now, I know I must save the shared_ptr in the archive myself to have my
own reference and to get all weak_ptr running properly. But I came
across this use case and I can't take it out of my mind, so if you have
an explanation, I'm all ears.
Thank you
Boost-users list run by williamkempf at hotmail.com, kalb at libertysoft.com, bjorn.karlsson at readsoft.com, gregod at cs.rpi.edu, wekempf at cox.net