Подтвердить что ты не робот

Отключение лямбда-функций в Qt5

Можно ли отключить функцию лямбда? И если "да", как?

Согласно https://qt-project.org/wiki/New_Signal_Slot_Syntax Мне нужно использовать QMetaObject::Connection, который возвращается из метода QObject:: connect, но тогда как Я передаю этот объект функции лямбда?

Пример псевдокода:

QMetaObject::Connection conn = QObject::connect(m_sock, &QLocalSocket::readyRead, [this](){
    QObject::disconnect(conn); //<---- Won't work because conn isn't captured

    //do some stuff with sock, like sock->readAll();
}
4b9b3361

Ответ 1

Если вы непосредственно захватываете conn, вы захватываете неинициализированный объект путем копирования, что приводит к неопределенному поведению. Вам нужно захватить умный указатель:

std::unique_ptr<QMetaObject::Connection> pconn{new QMetaObject::Connection};
QMetaObject::Connection &conn = *pconn;
conn = QObject::connect(m_sock, &QLocalSocket::readyRead, [this, pconn, &conn](){
    QObject::disconnect(conn);
    // ...
}

Или с помощью общего указателя с немного большими накладными расходами:

auto conn = std::make_shared<QMetaObject::Connection>();
*conn = QObject::connect(m_sock, &QLocalSocket::readyRead, [this, conn](){
    QObject::disconnect(*conn);
    // ...
}

Начиная с Qt 5.2 вы можете использовать объект контекста:

std::unique_ptr<QObject> context{new QObject};
QObject* pcontext = context.get();
QObject::connect(m_sock, &QLocalSocket::readyRead, pcontext,
    [this, context = std::move(context)]() mutable {
    context.release();
        // ...
 });

Ответ 2

Контекстное решение из ответа ecatmur - самый простой вариант, но я думаю, что использование умного указателя усложняет понимание. Вместо этого я бы использовал необработанный указатель:

QObject *context = new QObject(this);
connect(sender, &Sender::signal, context, [context] {
  delete context;
  // ...
});