T
Techstella25. Januar 2021 02:21

AbstractListModel использование в нескольких элементах

Всем привет! Подскажите почему у меня крашится программа когда я использую мою абстрактную модель в повторяющихся элементах QML?
У меня есть 3 элемента (2 из них одинаковые) которые используют одну и ту же модель данных. Когда ко всем трём присвоена модель, программа крашится, иногда выдаёт std::bad_alloc. Крашится тока в дебаге, в релизе работает. Когда от одного из повторяющихся снимаю модель тогда работает.

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.

Magst du es? In sozialen Netzwerken teilen!

8
Алексей Внуков
  • 25. Januar 2021 03:11

без кода - годание на кофейной гуще.

    T
    • 25. Januar 2021 04:05
    • (bearbeitet)

    Не знаю чем поможет код. Смымсл в том что есть listView Arm, на экране их 2, они должны отображать данные одной и той же модели. Но программа крашится когда присваеваешь модель обоимю

    // main
    qmlRegisterType<DriveModel>("TS.Drive", 1, 0, "DriveModel");
    qmlRegisterUncreatableType<Drives>("TS.Drive", 1, 0, "Drives", "");
    qmlRegisterUncreatableMetaObject(DriveModel::staticMetaObject, "TS.Drive", 1, 0, "Drive", "Error: only enums");
    context->setContextProperty("drives", &drives);
    
    // model
    class Drives;
    class DriveModel : public QAbstractListModel
    {
        Q_OBJECT
        Q_PROPERTY(Drives *list READ list WRITE setList)
    
    public:
        explicit DriveModel(QObject *parent = nullptr) : QAbstractListModel(parent) , mList(nullptr){ }
        enum Role
        {
            TYPE = 0,
            CTRL = 1,
            MOTOR = 2,
            SENSOR = 3,
            MOTQTY = 4,
        };
        Q_ENUM(Role)
    
    
        int rowCount(const QModelIndex &parent = QModelIndex()) const override
        {
            if (parent.isValid() || !mList) return 0;
            return mList->size();
        }
    
        QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override
        {
            if (!index.isValid() || !mList) return QVariant();
            const Drive &drive = mList->at(index.row());
            switch (role)
            {
            case TYPE:return QVariant(drive.type);
            case CTRL:return QVariant(drive.ctrl);
            case MOTOR:return QVariant(drive.motor);
            case SENSOR:return QVariant(drive.sensor);
            case MOTQTY:return QVariant(drive.motorQty);
            }
            return QVariant();
        }
    
        virtual QHash<int, QByteArray> roleNames() const override
        {
            QHash<int, QByteArray> names;
            names[TYPE] = "type";
            names[CTRL] = "ctrl";
            names[MOTOR] = "motor";
            names[SENSOR] = "sensor";
            names[MOTQTY] = "motQty";
            return names;
        }
    
        Drives *list() const
        {
            return mList;
        }
    
        void setList(Drives *list)
        {
            beginResetModel();
            if (mList) mList->disconnect(this);
            mList = list;
    
            if (mList) {
                connect(mList, &Drives::preItemAppended, this, [=]() {
                    const int index = mList->size();
                    beginInsertRows(QModelIndex(), index, index);
                });
                connect(mList, &Drives::postItemAppended, this, [=]() {
                    endInsertRows();
                });
    
                connect(mList, &Drives::preItemRemoved, this, [=](int index) {
                    beginRemoveRows(QModelIndex(), index, index);
                });
                connect(mList, &Drives::postItemRemoved, this, [=]() {
                    endRemoveRows();
                });
                connect(mList, &Drives::preResetting, this, [=]() {
                    beginResetModel();
                });
                connect(mList, &Drives::postResetting, this, [=]() {
                    endResetModel();
                });
                connect(mList, &Drives::dataChanged, this, [=](int index, int role) {
                    QModelIndex i = createIndex(index, 0);
                    emit dataChanged(i, i, QVector<int>() << role);
                });
            }
            endResetModel();
        }
    
    private:
        Drives *mList;
    };
    
    DriveModel{
        id: drivesModel
        list: drives
    }
    
    Item {
        height: 790
        width: 260
        ArmBackground{ id: leftArmBgrd; position: 1;}
        ColumnLayout{
            anchors.fill: parent
            StyleLabel{
                text: qsTr("ЛЕВЫЙ РЫЧАГ")
                Layout.alignment: Qt.AlignHCenter
                Layout.topMargin: 10
                color: leftArmColor;
            }
            Arm{ id: leftArm; position: 1;// model: drivesModel
                Layout.fillHeight: true; Layout.leftMargin: 5 }
            JoystickEmulator{
                id:armEmul
                width: parent.width - 80
                height: width
                Layout.leftMargin: 30
                onMoved: ctrl.onUpdateVirtualArm(xValue, yValue,
                                                 armEmul1.xValue, armEmul1.yValue);
            }
            Item { height: 10;}
        }
    }
    
    Item {
        height: 790
        width: 260
        ArmBackground{id: rightArmBgrd; position: 2}
        ColumnLayout{
            anchors.fill: parent
            StyleLabel{
                text: qsTr("ПРАВЫЙ РЫЧАГ")
                Layout.alignment: Qt.AlignHCenter
                Layout.topMargin: 10
                color: rightArmColor;
            }
            Arm{ id: rightArm; position: 2; model: drivesModel
                Layout.fillHeight: true; Layout.leftMargin: 25 }
            JoystickEmulator{
                id:armEmul1
                width: parent.width - 80
                height: width
                Layout.leftMargin: 50
                onMoved: ctrl.onUpdateVirtualArm(armEmul.xValue,
                                                 armEmul.yValue, xValue, yValue);
            }
            Item { height: 10;}
        }
    }
    
    

    QObject::~QObject: Timers cannot be stopped from another thread
    QML debugging is enabled. Only use this in a safe environment.
    terminate called after throwing an instance of 'std::bad_alloc'
    what(): std::bad_alloc

      T
      • 25. Januar 2021 04:19

      Да и ещё есть третий элемент, он свёрстан подругому нежели эти, но тоже отображает эту модель. И вот он никак не влияет. Падает только когда 2 одинаковых элемента с одной моделью. Неужели под каждый отдельную модель делать?

        Алексей Внуков
        • 25. Januar 2021 05:44
        • (bearbeitet)

        ну начнем с того что у вас не правильно организована работа с потоками, о чем говорит первое предупреждение.
        у вас так же где-то не правильно создается обьект, из-за чего выскакивает std::bad_alloc. что за элемент Arm? для примера у меня могут и 10 элементов одновременно в одном окне использовать одну и туже модель без ошибок.
        проблема где-то у вас в реализации

          T
          • 25. Januar 2021 07:42
          • Die Antwort wurde als Lösung markiert.

          Разобрался. Извиняюсь за беспокойство. Я просто не поставил "model." перед переменными. Делаю рефакторинг под abstractModel, раньше модель была в QML и как то прокатывала без этого. На железе уже пол года работает старая версия. С потоками и bad alloc пока не понятно, ведь до abstractModel проблем не было и сейчас нет как проставил model. везде.

            T
            • 29. Januar 2021 04:15

            К сожалению проблема всплыла позже. Если у вас есть несколько QAbstractModel то важно чтоб в них не было одинаковых так называемых rolenames. У меня в разных моделях было "type", из за этого она всё время била по неинициализованной памяти и падала.

              Алексей Внуков
              • 29. Januar 2021 06:51
              • (bearbeitet)

              разные модели могут иметь одинаковые rolenames, и это никак не влияет на их работу. самый тривиальный пример - это модели которые отображают данные из БД, у каждой второй таблици есть поле id или n, и везде будет корректно работать роль idRole или nRole. проблемы могут быть в неправильной реализации модели

                T
                • 5. Februar 2021 04:27

                Не прав был. В общем 3 дня разбирался. В итоге отказался от абстрактной модели, так как посмотрел на количество вызовов data(), на модель из 50ти параметров это было очень много при даже 10ти элементах. И даже с моделью созданой в qml программа вылетала на отладке. Особенно сразу падала при быстрой перемотке ListView. Создал 3 отдельные модели, тока при таких условиях не падает.

                  Kommentare

                  Nur autorisierte Benutzer können Kommentare posten.
                  Bitte Anmelden oder Registrieren
                  Letzte Kommentare
                  ИМ
                  Игорь Максимов5. Oktober 2024 07:51
                  Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                  d
                  dblas55. Juli 2024 11:02
                  QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                  k
                  kmssr8. Februar 2024 18:43
                  Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                  Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                  EVA
                  EVA25. Dezember 2023 10:30
                  Boost - statisches Verknüpfen im CMake-Projekt unter Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
                  Jetzt im Forum diskutieren
                  J
                  JacobFib17. Oktober 2024 03:27
                  добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
                  JW
                  Jhon Wick1. Oktober 2024 15:52
                  Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
                  КГ
                  Кирилл Гусарев27. September 2024 09:09
                  Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
                  F
                  Fynjy22. Juli 2024 04:15
                  при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

                  Folgen Sie uns in sozialen Netzwerken