Фильтрация контейнера ( std::vector ) std::shared_ptr в контейнер std::weak_ptr

Я пытаюсь отфильтровать контейнер shared_ptr и сохранить отфильтрованный контент в контейнере, которому не принадлежит (weak_ptr). Найденная ниже программа дает сбой. Может ли кто-нибудь увидеть, что мне не хватает?

    #include <memory>
    #include <iostream>
    #include <vector>
    #include <algorithm>
    #include <functional>

    #include <boost/bind.hpp>

    struct A
    {
        int a_val;

        explicit A():a_val(0)           { std::cout << "A default constructed\n"; }
        explicit A(int a):a_val(a)      { std::cout << "A argument constructed\n";  }
        A(const A& other)               { a_val = other.a_val ; std::cout << "A copy constructed\n";  } 
        A& operator=(const A& other)    { a_val = other.a_val ; std::cout << "A copy assigned\n";   return *this; } 
        ~A()                            { std::cout << "A is destroyed\n"; }



        struct a_visitor
        {
            std::vector < std::weak_ptr < A >  > filtered_a;

            a_visitor() { filtered_a.resize(0); }

            void operator() (std::shared_ptr < A > a_ref)
            {
                if(a_ref->a_val % 2 )
                    filtered_a.push_back(a_ref);
            }
        };

        std::function < void ( std::shared_ptr < A > ) > a_visit_function;


        void visit_vector ( a_visitor *visitor)
        {
            a_visit_function = boost::bind(&a_visitor::operator(), visitor, _1);
            std::shared_ptr< A > current_node(this);
            a_visit_function(current_node);
        }

    };

    int main(void)
    {

        std::vector < std::shared_ptr < A > > a_array;

        for(int i=10;i<20;i++)
        {
            a_array.emplace_back(std::make_shared<A>(A(i)));
        }
        std::cout << "--------------------------\n";
        std::for_each(a_array.begin(), a_array.end(), ([&]( std::shared_ptr<A> &a_ref){
            std::cout << a_ref << " : " << a_ref->a_val << std::endl;
        }));
        std::cout << "--------------------------\n";

        A::a_visitor visitor;
        std::for_each(a_array.begin(), a_array.end(), ([&](std::shared_ptr < A > a_ref){
            a_ref->visit_vector(&visitor);
        }));

        std::cout << "--------------------------\n";
        std::for_each(visitor.filtered_a.begin(), visitor.filtered_a.end(), ([&](std::weak_ptr<A> &a_ref){
            std::cout << a_ref.lock()->a_val << std::endl;
        }));
        std::cout << "--------------------------\n";
    }

person balas bellobas    schedule 15.05.2014    source источник


Ответы (1)


Вот в чем проблема: std::shared_ptr< A > current_node(this);. Вы просто не можете сделать это по умолчанию, this имеет собственное время жизни. См.: std::shared_ptr этого

person Anycorn    schedule 15.05.2014
comment
спасибо, что указали мне на enable_shared_for_this. я не знал этого раньше. текущий дизайн выглядит хорошо. я пытаюсь иметь посетителя, который содержит не владеющие ссылки на объекты в контейнере. какие-либо комментарии по этому поводу? - person balas bellobas; 15.05.2014
comment
что, если вы посетите shared_ptr вместо посещенного вектора shared_ptr? а затем вызвать как `a_visitor посетитель; for_each(array.begin(), array.end(), посетитель); - person Anycorn; 15.05.2014
comment
Я не уверен, что получу ваше предложение @Anycorn. Не могли бы вы добавить больше? - person balas bellobas; 15.05.2014
comment
@balasbellobas В вашем дизайне объект A — это тот, кто посещает. Измените свой дизайн так, чтобы Объект был посещаемым. - person Anycorn; 15.05.2014