Обработка ошибок Qt / C++
Я провел много исследований по обработке ошибок с Qt / C++, и я все еще так же потерян, как и в начале. Возможно, я ищу простой выход (как и другие языки). Один, в частности, предусматривает необработанное исключение, которое я использую религиозно. Когда программа сталкивается с проблемой, она выдает необработанное исключение, чтобы я мог создать свой собственный отчет об ошибке. Этот отчет отправляется с моей машины клиентов на сервер в интернете, который я затем прочитал позже.
проблема, с которой я сталкиваюсь с C++, заключается в том, что любая обработка ошибок, которая сделана, должна быть продумана до руки (подумайте, попробуйте/поймайте или массивные условные обозначения). По моему опыту, проблемы в коде не рассматриваются раньше, чем рука, иначе не было бы проблемы с самого начала.
написание кросс-платформенного приложения без кросс-платформенной обработки ошибок/отчетности / механизма трассировки немного пугает меня.
мой вопрос: есть ли в Qt или C++ Конкретный механизм захвата ошибок "catch-all", который я могу использовать в своем приложении, чтобы, если что-то пойдет не так, я мог, по крайней мере, написать отчет до его сбоя?
пример:
class MainWindow: public QMainWindow
{
[...]
public slots:
void add_clicked();
}
void MainWindow::add_clicked()
{
QFileDialog dlg(this, Qt::Sheet);
QString filename = dlg.getOpenFileName(this);
if(!filename.isEmpty())
{
QStringList path = filename.split(QDir::separator());
QString file = path.at(path.count()); // Index out of range assertion.
if(!lst_tables->openDatabase(filename))
{
[...]
}
}
}
Я хочу, чтобы эта ошибка была поймана как необработанное исключение и приложение, чтобы выйти, не показывая пользователю окно сбоя по умолчанию в операционной системе Windows/Mac. Я просто хочу, чтобы он ушел красиво после записи сообщения утверждения в файл и т. д.
4 ответов
переопределить QCoreApplication::оповестить() и добавьте try-catch там. Это, и что-то в main() охватывает большинство случаев в моем опыте.
вот как я это делаю. Обратите внимание, что я использую C++ RTTI здесь, а не версию Qt, но это просто для удобства в наших приложениях. Кроме того, мы разместили QMessageBox с информацией и ссылкой на наш лог-файл. Вы должны расширяться в соответствии со своими потребностями.
bool QMyApplication::notify(QObject* receiver, QEvent* even)
{
try {
return QApplication::notify(receiver, event);
} catch (std::exception &e) {
qFatal("Error %s sending event %s to object %s (%s)",
e.what(), typeid(*event).name(), qPrintable(receiver->objectName()),
typeid(*receiver).name());
} catch (...) {
qFatal("Error <unknown> sending event %s to object %s (%s)",
typeid(*event).name(), qPrintable(receiver->objectName()),
typeid(*receiver).name());
}
// qFatal aborts, so this isn't really necessary
// but you might continue if you use a different logging lib
return false;
}
кроме того, мы используем __try, __кроме Windows для поймать asyncronous исключения (нарушения доступа). Google Breakpad, вероятно, может служить кросс-платформенной заменой для этого.
вы можете поставить ловушку (...) в функции main() вот здесь:
int main() try
{
...
}
catch (std::exception & e)
{
// do something with what...
}
catch (...)
{
// someone threw something undecypherable
}
Google Breakpad-это кросс-платформенная платформа отчетов об ошибках приложений. Может, это поможет?
(Я еще не пробовал это в наших приложениях C++/qt, но я бы хотел когда-нибудь это сделать...)
Qt обычно не использует или полностью поддерживает исключение (если вы можете в это поверить!)
проверьте эти ссылки:
почему Qt не использует обработку исключений?
http://doc.qt.io/qt-5/exceptionsafety.html
тем не менее, ответы от @Crazy Eddie и @Macke довольно хороши, но не всегда работают. В частности, я обнаружил, что вы не можете использовать ни один из них из функции слота, которую вы вызвали из В QML. Итак, я создал хакерскую работу для этой проблемы. * Используйте это в сочетании с их-не вместо него.
во-первых, я создал класс, производный от QException, который я пропущу здесь, но это то, что вы, вероятно, захотите сделать. В этом посте я просто называю его "MyQException".
в любом случае, добавьте этот заголовок для класса под названием QmlSlotThrower
:
#ifndef QMLSLOTTHROWER_H
#define QMLSLOTTHROWER_H
#include "MyQException.h"
class QmlSlotThrower
{
public:
static QmlSlotThrower *get()
{
static QmlSlotThrower instance;
return &instance;
}
QmlSlotThrower( QmlSlotThrower const& ) = delete;
void operator=( QmlSlotThrower const& ) = delete;
void throwToTop( const MyQException &exception );
private:
QmlSlotThrower(){}
};
static QmlSlotThrower *qmlSlotThrower = QmlSlotThrower::get();
#define throwFromQmlSlot( exc ) qmlSlotThrower->throwToTop( exc ); return;
#endif // QMLSLOTTHROWER_H
тогда это cpp:
#include "QmlSlotThrower.h"
#include <QTimer>
class AsynchronousThrower: public QObject
{
Q_OBJECT
public:
void throwThis( const MyQException &exception )
{
exception_ = exception;
QTimer::singleShot( 0, this, SLOT( throwIt() ) );
}
private slots:
void throwIt(){ throw exception_; }
private:
MyQException exception_;
};
static AsynchronousThrower asycnThrower;
// This is needed to allow the Q_OBJECT macro
// to work in the private classes
#include "QmlSlotThrower.moc"
// --------------------------------
void QmlSlotThrower::throwToTop( const MyQException &exception )
{ asycnThrower.throwThis( exception ); }
наконец, вот пример реализация:
void someQMLSlot()
{
// Qt has been progressively adding exception handling
// support, but you still cannot throw from a QML
// triggered slot. It causes an uncatchable fatal error!
// As a general rule, don't throw in Qt unless you are
// certain something is there to catch it. You cannot
// count on an uncaught exception handler at a top level
// to always work. This QML problem is a perfect example.
// So this is not an option here!
//throw MyQException( "Something terrible occured!" );
// This work around, however, can be used instead!
//throwFromQmlSlot( MyQException( "Something terrible occured!" ) )
// Or, to be more robust in illustrating how you can still use
// normal throws from nested functions even, you can do this:
try{ throw MyQException( "Something terrible occured!" ); }
catch( const MyQException &e) { throwFromQmlSlot( e ) }
qDebug() << "YOU SHOULD NEVER SEE THIS!!";
}
ИСПОЛЬЗУЙТЕ ТОЛЬКО МАКРОС НЕПОСРЕДСТВЕННО ИЗ СВОЕГО СЛОТА!