我有一个Request_manager类,其中有嵌套类:

class Request_manager {
    public:
        class Ticket {
        protected:
            explicit Ticket(System_time start_time);
            virtual void start() = 0;
        }
        struct ticket_element_{
            int app;
            std::shared_ptr <Ticket> ticket;
        };
    
        class Tickets_Queue{
        private:
            std::queue <ticket_element_> m_queue;
    
        public:
            void push(ticket_element_ ticket);
        };
    };

Request_manager::TicketTicket等其他类的父类,它们具有start()的自己的实现:

class Ticket : public Request_manager::Ticket,public std::enable_shared_from_this<Ticket> {
    void start();
};

ticket_element_可以保存来自Request_manager::Ticket的其他子级。在Request_manager::Tickets_Queue::push的实现中,我做到了:

void Request_manager::Tickets_Queue::push(ticket_element_ ticket_container){
    run_async(ticket_container.app,async::Policy::FIFO,[ticket_container]() { ticket_container.ticket->start(); });
}

我得到了错误:

In file included from data_interfaces/Request_manager.cpp:1:0:
data_interfaces/Request_manager.h: In lambda function:
data_interfaces/Request_manager.h:31:22: error: ‘virtual void dataserver::Request_manager::Ticket::start()’ is protected
         virtual void start() = 0;
                      ^
data_interfaces/Request_manager.cpp:66:67: error: within this context
             [ticket_container]() { ticket_container.ticket->start(); });

我不理解,因为一旦有了对象,我应该能够访问受保护的方法

iCMS 回答:类

ticket_container.ticket->start();中,您尝试从共享指针访问受保护的方法,该共享方法在基类Request_manager::Ticket中无权访问它,因此无法应用vtable

如果您在protected中将public更改为Request_manager::Ticket,它将可以访问方法并应用vtable,从而调用派生的Ticket

我已经添加/更改了一些内容以便能够复制:

class Request_manager
{
public:
    class Ticket {
    protected:
        explicit Ticket() {}
        ~Ticket() {}
    public:
        virtual void start() = 0;
    };
    struct ticket_element_{
        int app;
        std::shared_ptr <Ticket> ticket;
    };
    
    class Tickets_Queue{
    private:
        std::queue <ticket_element_> m_queue;
    
    public:
        void push(ticket_element_ ticket);
    };

    Tickets_Queue _queue;
};

class Ticket : public Request_manager::Ticket,public std::enable_shared_from_this<Ticket>
{
public:
    void start() {
        cout << "Starting ticket" << endl;
    }
};


void Request_manager::Tickets_Queue::push(ticket_element_ ticket_container){
    ticket_container.ticket->start();
}


int main() {
    Request_manager rm;

    rm._queue.push ({0,make_shared<Ticket>()});
    
    return 0;
}
manuel@desktop:~/projects$ g++ -Wall -Wextra -g main.cc -o main --std=gnu++2a -Wpedantic && time ./main
Starting ticket
本文链接:https://www.f2er.com/1918382.html

大家都在问