Back to TILs

C++ shared_ptr with custom deleter

Date: 2023-01-13Last modified: 2025-01-12

Table of contents

The purpose of Foo class is to trace the constructor and destructor calls.

struct Foo {
  int id{0};
  Foo(int i = 0) : id{i} { std::cout << "✅ Foo::Foo(" << i << ")\n"; }
  ~Foo() { std::cout << "❌ Foo::~Foo(), id=" << id << '\n'; }
};

The DeleterFunctor is a custom functor that will be used to release the objects when it goes out of context.

struct DeleterFunctor {
  void operator()(Foo* p) const {
    std::cout << "❌ Call delete from function object. Foo::id=" << p->id
              << '\n';
    delete p;
  }
};

This function wrap a POSIX fclose function call. It is used for trace purpose only.

int myfclose(FILE* stream) {
  std::cout << "❌ myfclose called\n";
  return fclose(stream);
}

Context block 1

  {
    std::cout << "1) constructor with no managed object\n";
    std::shared_ptr<Foo> sh1;
    if (sh1 == nullptr) {
      std::cout << "sh1 == nullptr\n";
    } else {
      std::cout << "sh1.id = " << sh1->id << '\n';  // never called
    }
    std::cout << "==> Context block 1 is going out of context here\n";
  }

Output of context block 1

1) constructor with no managed object
sh1 == nullptr
==> Context block 1 is going out of context here

Context block 2

  {
    std::cout << "\n2) constructor with object\n";
    std::shared_ptr<Foo> sh2(new Foo{10});
    std::cout << "sh2.id = " << sh2->id << '\n';
    std::cout << "sh2.use_count(): " << sh2.use_count() << '\n';
    std::shared_ptr<Foo> sh3(sh2);
    std::cout << "sh3.id = " << sh3->id << '\n';
    std::cout << "sh2.use_count(): " << sh2.use_count() << '\n';
    std::cout << "sh3.use_count(): " << sh3.use_count() << '\n';

    sh2->id = 123;
    std::cout << "sh3.id = " << sh3->id << '\n';
    std::cout << "==> Context block 2 is going out of context here\n";
  }

Output of context block 2

2) constructor with object
✅ Foo::Foo(10)
sh2.id = 10
sh2.use_count(): 1
sh3.id = 10
sh2.use_count(): 2
sh3.use_count(): 2
sh3.id = 123
==> Context block 2 is going out of context here
❌ Foo::~Foo(), id=123

Context block 3

  {
    std::cout << "\n3) constructor with object and deleter\n";
    std::shared_ptr<Foo> sh4(new Foo{11}, DeleterFunctor());
    std::shared_ptr<Foo> sh5(new Foo{12}, [](auto p) {
      std::cout << "Call delete from lambda... p->id=" << p->id << "\n";
      delete p;
    });
    std::cout << "==> Context block 3 is going out of context here\n";
  }

Output of context block 3

3) constructor with object and deleter
✅ Foo::Foo(11)
✅ Foo::Foo(12)
==> Context block 3 is going out of context here
Call delete from lambda... p->id=12
❌ Foo::~Foo(), id=12
❌ Call delete from function object. Foo::id=11
❌ Foo::~Foo(), id=11

Context block 4

  {
    std::cout << "\n4) constructor with pipe object and deleter (POSIX)\n";
    // Borrowed from GrUtils::exec()
    std::array<char, 128> buffer;
    std::string result;
    std::shared_ptr<FILE> pipe(popen("ls -l shared_ptr_with_deleter.cpp", "r"),
                               pclose);  // Flawfinder: ignore

    if (!pipe) {
      throw std::runtime_error("popen() failed!");
    }

    while (!feof(pipe.get())) {
      if (fgets(buffer.data(), 128, pipe.get()) != nullptr)
        result += buffer.data();
    }
    std::cout << "ls result: " << result << '\n';
    std::cout << "==> Context block 4 is going out of context here\n";
  }

Output of context block 4

4) constructor with pipe object and deleter (POSIX)
ls result: -rw-r--r-- 1 geraldo geraldo 4838 Jan 27  2024 shared_ptr_with_deleter.cpp

==> Context block 4 is going out of context here

Context block 5

  {
    std::cout << "\n5) constructor with pipe object and deleter (POSIX)\n";
    std::shared_ptr<FILE> file1(fopen("/tmp/test1.txt", "w"), fclose);
    std::shared_ptr<FILE> file2(fopen("/tmp/test2.txt", "w"), myfclose);
    std::string s{"Some text"};
    fwrite(s.c_str(), s.size(), 1, file1.get());
    fwrite(s.c_str(), s.size(), 1, file2.get());
    std::cout << "==> Context block 5 is going out of context here\n";
  }

Output of context block 5

5) constructor with pipe object and deleter (POSIX)
==> Context block 5 is going out of context here
❌ myfclose called

Possible output

1) constructor with no managed object
sh1 == nullptr
==> Context block 1 is going out of context here

2) constructor with object
✅ Foo::Foo(10)
sh2.id = 10
sh2.use_count(): 1
sh3.id = 10
sh2.use_count(): 2
sh3.use_count(): 2
sh3.id = 123
==> Context block 2 is going out of context here
❌ Foo::~Foo(), id=123

3) constructor with object and deleter
✅ Foo::Foo(11)
✅ Foo::Foo(12)
==> Context block 3 is going out of context here
Call delete from lambda... p->id=12
❌ Foo::~Foo(), id=12
❌ Call delete from function object. Foo::id=11
❌ Foo::~Foo(), id=11

4) constructor with pipe object and deleter (POSIX)
ls result: -rw-r--r-- 1 geraldo geraldo 4838 Jan 27  2024 shared_ptr_with_deleter.cpp

==> Context block 4 is going out of context here

5) constructor with pipe object and deleter (POSIX)
==> Context block 5 is going out of context here
❌ myfclose called

Content of /tmp/test1.txt

Some text

Content of /tmp/test2.txt

Some text

References