Evgenii Legotckoi
Evgenii LegotckoiFeb. 2, 2016, 11:10 a.m.

QML - Lesson 023. The hunt for bugs in the transmission pointer to QObject in QML

One of the most vile and little predictable bugs are those that arise in an uncertain time. Among these include a bug that appears when the pointer is transmitted to the QObject in QML layer. The problem is that if there is no parent QObject, the transmission layer occurs in QML object change of ownership, that is, it is set JavaScriptOwnership. As a result, when the QML layer lost all references to this object, it will be removed garbage collector QML. Accordingly, all references in C++ layer will be non-valid. And the app when trying to appeal to these links will collapse in silence, without saying anything about the cause of the collapse.

Speaking of response uncertainty bug, it should be noted that this uncertainty stems from the time of garbage collection. In a typical embodiment, the time may come when the program a couple of gigabytes of memory dine or when the memory is not enough, and it will be available by performing garbage collection. That is, the application can run without problems for a long time without causing a bug, and it is even possible to go through several versions before the bug will manifest itself at any user who decides to report it to the developers.

It is worth noting that this rule will not be applied to the objects declared as Q_PROPERTY .

But we can use QML function gc() to show the problem, which will accelerate the garbage collection.


Hunting for bugs

Now let's see how it is implemented in the code.

Project structure

In this project, we are interested in the following files:

  • main.cpp - do not get to pass the test object in QML layer without editing;
  • MyClass.h - header file for class production problems;
  • MyClass.cpp - file source code for class production problems;
  • main.qml - QML layer file that will destroy the object pointer.

MyClass.h

In the test class will be created using the object method createObject() , and used via useObject() method, respectively. Once in QML layer will be zeroed pointer and garbage collection will be made, then useObject() method puts the program.

#ifndef MYCLASS_H
#define MYCLASS_H

#include <QObject>

class MyClass : public QObject
{
    Q_OBJECT

public:
    Q_INVOKABLE QObject* createObject();    // Create an object to test
    Q_INVOKABLE void useObject();           // Using the test object

private:
    QObject* m_object;     
};

#endif // MYCLASS_H

MyClass.cpp

In the method createObject() there are two version of the initialization m_object object. The first option is to not comment out a bug, the second will work stably. And useObject() method will receive the name of the object, of course, that in this case we get an empty string.

#include "MyClass.h"
#include <QObject>
#include <QDebug>
#include <QString>

QObject *MyClass::createObject()
{
    m_object = new QObject;         // Create an object with no parent
//    m_object = new QObject(this);   // Create an object with the parent
    return m_object;                // return the object
}

void MyClass::useObject()
{
    qDebug() << m_object->objectName(); // Display name of the object
}

main.cpp

Standard object registration for access to the QML context.

#include <QApplication>
#include <QQmlApplicationEngine>
#include <QQmlContext>
#include "MyClass.h"

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);

    MyClass mClass;

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));

    // Register a test class in QML layer showing the problem
    engine.rootContext()->setContextProperty("myClass", &mClass);

    return app.exec();
}

main.qml

The project created three buttons:

  1. It will create the object and pass it a pointer to the property objectFromCppWorld ;
  2. Reset the index by creating a bug;
  3. Try to use a C++ object layer. And the method will work until the garbage collection will occur.
import QtQuick 2.5
import QtQuick.Controls 1.3

ApplicationWindow {
    width: 640
    height: 480
    visible: true

    property QtObject objectFromCppWorld: null

    /* To demonstrate the problem with the change of the parent object make 
     * three buttons that will activate in sequence. 
     * First create an object in C ++ layer and pass a pointer to it in QML layer. 
     * A second pointer will reset and run garbage collection. 
     * Third try to use an object
     */
    Column {
        anchors.centerIn: parent
        Button {
            text: "create object in C++, save the pointer to it in C++ world and pass it to QML"
            onClicked: objectFromCppWorld = myClass.createObject()
        }
        Button {
            text: "null the reference in QML"
            onClicked: {
                objectFromCppWorld = null
                gc()
            }
        }
        Button {
            // When garbage collection occurs, without the use of the parent object will not work
            text: "use created object in C++"
            onClicked: myClass.useObject()
        }
    }
}

Co-author of the article: Vladimir Kurman

Video

We recommend hosting TIMEWEB
We recommend hosting TIMEWEB
Stable hosting, on which the social network EVILEG is located. For projects on Django we recommend VDS hosting.

Do you like it? Share on social networks!

Comments

Only authorized users can post comments.
Please, Log in or Sign up
г
  • ги
  • April 23, 2024, 3:51 p.m.

C++ - Test 005. Structures and Classes

  • Result:41points,
  • Rating points-8
l
  • laei
  • April 23, 2024, 9:19 a.m.

C ++ - Test 004. Pointers, Arrays and Loops

  • Result:10points,
  • Rating points-10
l
  • laei
  • April 23, 2024, 9:17 a.m.

C++ - Тест 003. Условия и циклы

  • Result:50points,
  • Rating points-4
Last comments
k
kmssrFeb. 8, 2024, 6:43 p.m.
Qt Linux - Lesson 001. Autorun Qt application under Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
Qt WinAPI - Lesson 007. Working with ICMP Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
EVA
EVADec. 25, 2023, 10:30 a.m.
Boost - static linking in CMake project under Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
J
JonnyJoDec. 25, 2023, 8:38 a.m.
Boost - static linking in CMake project under Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
G
GvozdikDec. 18, 2023, 9:01 p.m.
Qt/C++ - Lesson 056. Connecting the Boost library in Qt for MinGW and MSVC compilers Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
Now discuss on the forum
G
GarApril 22, 2024, 5:46 a.m.
Clipboard Как скопировать окно целиком в clipb?
DA
Dr Gangil AcademicsApril 20, 2024, 7:45 a.m.
Unlock Your Aesthetic Potential: Explore MSC in Facial Aesthetics and Cosmetology in India Embark on a transformative journey with an msc in facial aesthetics and cosmetology in india . Delve into the intricate world of beauty and rejuvenation, guided by expert faculty and …
a
a_vlasovApril 14, 2024, 6:41 a.m.
Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
Павел Дорофеев
Павел ДорофеевApril 14, 2024, 2:35 a.m.
QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
f
fastrexApril 4, 2024, 4:47 a.m.
Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…

Follow us in social networks