Как этот shared_ptr автоматически конвертируется в необработанный указатель? - PullRequest
0 голосов
/ 29 января 2019

Я изучаю enable_shared_from_this C ++ 11 сейчас;один пример привел меня в замешательство: как тип shared_ptr, возвращаемый shared_from_this(), может преобразоваться в этот необработанный указатель?

#include <iostream>
#include <memory>
#include <functional>

struct Bar {
    Bar(int a) : a(a) {}
    int a;
};

struct Foo : public std::enable_shared_from_this<Foo> {
    Foo() { std::cout << "Foo::Foo\n"; }
    ~Foo() { std::cout << "Foo::~Foo\n"; }

    std::shared_ptr<Bar> getBar(int a)
    {
        std::shared_ptr<Bar> pb(
            new Bar{a}, std::bind(&Foo::showInfo, shared_from_this(), std::placeholders::_1)
        );
        return pb;
    }

    void showInfo(Bar *pb)
    {
        std::cout << "Foo::showInfo()\n";
        delete pb;
    }

};

int main()
{
    std::shared_ptr<Foo> pf(new Foo);
    std::shared_ptr<Bar> pb = pf->getBar(10);
    std::cout << "pf use_count: " << pf.use_count() << std::endl;
}

1 Ответ

0 голосов
/ 29 января 2019

Это std::bind, будучи умным, а не указателем.

Как описано в Callable, при вызове указателя на нестатическую функцию-член или указателя на нестандартныйчлен статических данных, первый аргумент должен быть ссылкой или указателем (включая, возможно, умный указатель, такой как std :: shared_ptr и std :: unique_ptr) на объект, к которому будет получен доступ.

* Реализуется 1010 *, поэтому он может принимать умный указатель вместо необработанного указателя.

В реализации glibc ++ вы можете видеть, что bind внутренне вызывает invoke:

  // Call unqualified
  template<typename _Result, typename... _Args, std::size_t... _Indexes>
_Result
__call(tuple<_Args...>&& __args, _Index_tuple<_Indexes...>)
{
  return std::__invoke(_M_f,
      _Mu<_Bound_args>()(std::get<_Indexes>(_M_bound_args), __args)...
      );
}

И std::invoke работает со смарт-вещами (указатели, справочные обертки и т. Д.) "Из коробки":

INVOKE(f, t1, t2, ..., tN) определяется следующим образом:

Если f является указателем на функцию-член класса T:

  • Если std::is_base_of<T, std::decay_t<decltype(t1)>>::value имеет значение true,тогда INVOKE(f, t1, t2, ..., tN) эквивалентно (t1.*f)(t2, ..., tN)
  • Если std::decay_t<decltype(t1)> является специализацией std::reference_wrapper, то INVOKE(f, t1, t2, ..., tN) эквивалентно (t1.get().*f)(t2, ..., tN)
  • Если t1 не удовлетворяет предыдущим пунктам, то INVOKE(f, t1, t2, ..., tN) эквивалентно ((*t1).*f)(t2, ..., tN).
...