Реклама
  • MinusNull
  • Статья
  • 15 января 2018 г. 10:18

Шаблон программирования Pimpl - то, что вам следует знать

C++, Pimpl

Основы

Вы можете встретить шаблон Pimpl под другими именами: d-pointer, compiler firewall или даже шаблон Cheshire Cat или непрозрачный указатель.

В его основной форме шаблон выглядит следующим образом:

  • В классе мы перемещаем все закрытые члены в новый объявленный тип, например, в класс PrivateImpl .
  • Объявляем PrivateImpl в заголовочном файле основного класса.
  • В соответствующем файле cpp объявляем класс PrivateImpl и определяем его.
  • Теперь, если вы измените закрытую реализацию, код клиента не будет перекомпилирован (поскольку интерфейс не изменился).

Таким образом, это может выглядеть так (грубый, старый стиль кода!):

class.h

class MyClassImpl;
class MyClass {
    // ...
    void Foo();
private:    
    MyClassImpl* m_pImpl; // Предупреждение!!! 
                          // raw-указатель! :)
};

class.cpp

class MyClassImpl
{
public:
    void DoStuff() { /*...*/ }
};

MyClass::MyClass () 
: m_pImpl(new MyClassImpl()) 
{ }

MyClass::~MyClass () { delete m_pImpl; }

void MyClass ::DoSth() {
    m_pImpl->DoSth();
}

Эх ... уродливые raw-указатели!

Итак, кратко: мы упаковываем все, что является закрытым, в этот объявленный наперед класс. Мы используем только один член нашего основного класса - компилятор может работать только с указателем без полного объявления типа, поскольку необходим только размер указателя. Затем объявление и реализация происходят в файле .cpp .

Конечно, в современном C ++ рекомендуется использовать unique_ptr , а не raw-указатели.

Два очевидных недостатка этого подхода: нам нужно выделение памяти для хранения закрытой секции. А также основной класс просто перенаправляет вызов метода на закрытую реализацию.

Хорошо ... но это все ... верно? Не так просто!

Вышеприведенный код может работать, но мы должны добавить несколько бит, чтобы он работал в реальной жизни.

  • BlinCT
  • Ответ
  • 15 января 2018 г. 7:28

Проблема добавления #DEFINE при сборке CMak'ом

Добрый день.
Попробуйте вот в таком порядке сделать, я не сколько раз натыкался на случай когда та или иная строка не принималась потому что не там была.

cmake_minimum_required(VERSION 3.5.1)
project(Gener095) include_directories(../../088/SharedCode) add_definitions(-DGENER095_LIBRARY -DFRIEND_CLASS=Gener095_TestClass) add_definitions(-DLIB_NAME=\"${PROJECT_NAME}\") set(VERSION_TARGET "0.1.0") set(SOURCE_FILES Messages/message_095_data.cpp Messages/message_095_data.h gener095_core.cpp gener095_core.h gener095_global.cpp gener095_global.h taremanager.cpp taremanager.h ) qt5_add_resources(RCC_RESOURCES gener095_resources.qrc) project_generate_version("${VERSION_TARGET}" "lib") add_library(${PROJECT_NAME} SHARED ${SOURCE_FILES} ${RCC_RESOURCES}) target_link_libraries(${PROJECT_NAME} PUBLIC Qt5::Xml) target_link_libraries(${PROJECT_NAME} PUBLIC Qt5::Gui) target_link_libraries(${PROJECT_NAME} PUBLIC SharedCode) target_include_directories(${PROJECT_NAME} PUBLIC ${CMAKE_CURRENT_SOURCE_DIR}/../../088) link_directories(${CMAKE_BINARY_DIR}/088/build-libs/SharedCode)
  • folax
  • Комментарий
  • 12 января 2018 г. 8:13

QML - Урок 021. Переключение между окнами в QML

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

    Logic logic;
    QQmlApplicationEngine engine;
    engine.rootContext()->setContextProperty("logic", &logic);
    engine.load(QUrl(QStringLiteral("qrc:/qml/main.qml")));

    return app.exec();
}

Добрый день, этот код не ваш, я видел этот код так как тоже его делал. Это задание для прохождения на собеседование в одну из крупных украинских IT компаний. Для переключения между окнами используйте Loader QML Type . И на будущее если используете чужой код, изменяйте его, так как не честно использовать чужое, или хотя бы указывайте что код не мой. Будут вопросы пишите.
  • Vladimir
  • Вопрос
  • 7 января 2018 г. 16:29

pyqt5 ввод чисел

Как ввести данные в программе python 3 pyqt5 ?


  • HelgeID
  • Ответ
  • 25 декабря 2017 г. 11:29

Сборка финального проекта Qt/QML

И так протестировал немного, что я заметил:
- собрал библиотеки при помощи windeployqt - но приложение qml quick не запускается все-равно (даже не видно процесса в диспетчере задач).
- решение этой проблемы было удаление Qt5Core.dll, но возникла новая проблема - приложение работает только на моем компьютере, на другом просит библиотеку Qt5Core.dll  - PARADOX .
- скопировал я папки QtQml и QtQuick.2 у папку с exe-файлом. (библиотеку Qt5Core.dll - оставил у папке с exe-файлом)
В результате все заработало, правда - это только тестовый проект.

Вопрос: Что теперь надо часть Qt копировать и кучу файлов с собой таскать, иначе никак, какие еще есть способы? О покупке лицензии на Qt знаю!
Ведь если делать статическую линковку - на другом компьютере не работает. А как это решить?

Сборка финального проекта Qt/QML

Если более подробно про динамическую сборку, то:

1. Соберите dll посредством windeployqt https://evileg.com/post/163/
2. Из папки вашего компилятора скопируйте папки QtQuick, QtQuick.2 в корневую папку где ваш exe. Удалите из них отладочные библиотеки (названия файлов заканчиваются буквой d).
3. Проверяйте на ПК без Qt.
Если приложение висит в процессах, но не запускается - проверьте библиотеки в корне с exe, которые собрал windeployqt. Дата их изменения должна соответствовать таким же библиотекам в папке вашего компилятора. У меня почему-то Qt5Core.dll попал от другого компилятора, помогла его замена.
  • HelgeID
  • Вопрос
  • 24 декабря 2017 г. 22:08

Сборка финального проекта Qt/QML

Привет ребята! У меня такая проблема: хочу собрать проект на Qt/QML.
Вот например имею такой код:
main.cpp
#include <QGuiApplication>
#include <QQmlApplicationEngine>

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

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
    if (engine.rootObjects().isEmpty())
        return -1;

    return app.exec();
}
main.qml
import QtQuick 2.5
import QtQuick.Controls 1.0

ApplicationWindow {
    id: root

    visible: true
    width: 500
    height: 300
    title: "QtWindow"

    Button {
        anchors.centerIn: parent
        text: "Push me"
        onClicked: root.color = "blue"
    }
}
MyQtQuick.pro
TEMPLATE = app
QT += qml quick
CONFIG += c++11
SOURCES += main.cpp
RESOURCES += qml.qrc
QML_IMPORT_PATH =
QML_DESIGNER_IMPORT_PATH =
DEFINES += QT_DEPRECATED_WARNINGS

qnx: target.path = /tmp/$${TARGET}/bin
else: unix:!android: target.path = /opt/$${TARGET}/bin
!isEmpty(target.path): INSTALLS += target

Суть проблемы в том, что хочу запустить без редактора или не в среде Qt (отдельно).
Оно хочет библиотек! Ну лады начал вытаскивать по одной:
Вот список (сразу для релиза):
Qt5Core.dll, Qt5Gui.dll, Qt5Network.dll, Qt5Qml.dll, libgcc_s_dw2-1.dll, libwinpthread-1.dll
и все больше ничего не требует. Приложение не работает так как надо. В диспетчере задач запускается процесс и сразу завершился! Ошибки никакой нет! Ну и как здесь быть? Это одна проблема!

Второе: Думаю, может сделать статическую линковку. Ну хорошо! Сделал все по статье: https://vladsol.tk/2017/05/qt-5-8-0-static-build/ ,  делало 12 часов :)

Потом в настройках(сборка и запуск) добавил профиль, комплекты и тд.
Ну вроде работает, но только на моем компьютере. Если перенести на другой компьютер, или виртуальную машину, то выскакивает ошибка приложения
(при этом ничего не требует).
Как решить такую задачу?

Вот некоторая инфа:
os : win7 SP1 32bit
компилятор : MinGW 32bit
ver Qt : 5.9.1
проект : qml quick

  • Миша
  • Ответ
  • 19 декабря 2017 г. 10:01

Код не работает после QEventLoop

после такой записи

connect (& futureWatcher, & QFutureWatcher<void>::finished, &loop, &QEventLoop::exit);
у меня появилось 3 ошибки:
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qglobal.h:738: ошибка: static assertion failed: The slot requires more arguments than the signal provides.
#define Q_STATIC_ASSERT_X(Condition, Message) static_assert(bool(Condition), Message)
^
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qglobal.h:738: ошибка: static assertion failed: Signal and slot arguments are not compatible.
#define Q_STATIC_ASSERT_X(Condition, Message) static_assert(bool(Condition), Message)
^
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobjectdefs_impl.h:74: ошибка: no type named 'Car' in 'struct QtPrivate::List<>'
typedef typename List_Append<List<typename L::Car>,typename List_Left<typename L::Cdr, N - 1>::Value>::Value Value;
^

  • EVILEG
  • Комментарий
  • 18 декабря 2017 г. 18:26

PyQt5 - Урок 005. Автодополнение для поля ввода с помощью QCompleter

Пожалуйста ))
Будут какие затруднения, задавайте вопросы на форуме в разделе PyQt5. PyQt5 не является моей рабочей нишей, но я работаю периодически с python, а сама библиотека Qt - это моё основное направление. Так что, если будет возможным, отвечу на вопросы или направлю, куда копать. Да и сообщество может поможет, ну и сами не стесняйтесь отвечать на вопросы других пользователей.

  • Миша
  • Ответ
  • 18 декабря 2017 г. 13:13

Пауза в многопоточности

C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});
^
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:185: ошибка: use of deleted function 'QFutureWatcherBase::QFutureWatcherBase(const QFutureWatcherBase&)'
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:454: ошибка: 'QObject::QObject(const QObject&)' is private
Q_DISABLE_COPY(QObject)
^
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:55: ошибка: within this context
class Q_CORE_EXPORT QFutureWatcherBase : public QObject
^
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qfuturewatcher.h:55: ошибка: use of deleted function 'QObject::QObject(const QObject&)'
C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>::<lambda>(MainWindow::on_ButtonRunMulti_clicked()::<lambda()>&&)'
connect (& futureWatcher, & QFutureWatcher <void> :: progressValueChanged, [=](){ui->ProgressBar2->setValue(25);qDebug()<<futureWatcher.progressValue();});
^
C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:307: ошибка: use of deleted function 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>::<lambda>(const MainWindow::on_ButtonRunMulti_clicked()::<lambda()>&)'
return connect(sender, signal, sender, slot, Qt::DirectConnection);
^
C:\Qt\project\MyCandleSrick\mainwindow.cpp:1116: ошибка: use of deleted function 'QFutureWatcher<void>::QFutureWatcher(const QFutureWatcher<void>&)'
C:\Qt\Qt5.9.2\5.9.2\mingw53_32\include\QtCore\qobject.h:313: ошибка: 'static typename std::enable_if<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::type QObject::connect(const typename QtPrivate::FunctionPointer<Func>::Object*, Func1, const QObject*, Func2, Qt::ConnectionType) [with Func1 = void (QFutureWatcherBase::*)(int); Func2 = MainWindow::on_ButtonRunMulti_clicked()::<lambda()>; typename std::enable_if<(QtPrivate::FunctionPointer<Func2>::ArgumentCount == (-1)), QMetaObject::Connection>::type = QMetaObject::Connection; typename QtPrivate::FunctionPointer<Func>::Object = QFutureWatcherBase]', declared using local type 'MainWindow::on_ButtonRunMulti_clicked()::<lambda()>', is used but never defined [-fpermissive]

Реклама
  • falcon
  • 16 января 2018 г. 17:25

Qt - Тест 001. Сигналы и слоты

  • Результат 100 баллов
  • Очки рейтинга 10
  • falcon
  • 16 января 2018 г. 17:22

Qt - Тест 001. Сигналы и слоты

  • Результат 68 баллов
  • Очки рейтинга -1
  • falcon
  • 16 января 2018 г. 17:18

C++ - Тест 001. Первая программа и типы данных

  • Результат 73 баллов
  • Очки рейтинга 1
Последние комментарии

QML - Урок 021. Переключение между окнами в QML

Спасибо всем. Все получилось. Прикручиваю логику.

  • BlinCT
  • 14 января 2018 г. 19:28

Разработка на Qt под iOS

Вот честно, на сколько же муторно под огрызок что то делать. Куча проблем) А вод линь или под Андроид все просто и тривиально))

  • folax
  • 12 января 2018 г. 9:16

QML - Урок 021. Переключение между окнами в QML

Ничего сложного, делаете по тех заданию 3 файла qml, называете их как указанно в тех задании, потом из первого окна через Loader их переключаете, в окне 2 и 3 делаете сигналы которые при закры...

QML - Урок 021. Переключение между окнами в QML

Все верно, я и не говорил что этот кусок кода лично мое произведение. Это тоже верно: Это задание для прохождения на собеседование в одну из крупных украинских IT компаний. Логику ...

  • folax
  • 12 января 2018 г. 8:13

QML - Урок 021. Переключение между окнами в QML

int main(int argc, char *argv[]){ QApplication app(argc, argv); Logic logic; QQmlApplicationEngine engine; engine.rootContext()->setContextProperty("logic", &logic)...

Сейчас обсуждают на форуме

ChartView. Отображение метки данных точки серии при наведении курсора

Спасибо большущее за советы! Все получилось через ScatterSeries. Методы remove() как-то сходу не дались, удаляет в первый раз, а потом программа падает... Не стал тратить время и воспользовалс...

  • EVILEG
  • 16 января 2018 г. 14:23

Как проверить доступность сервера

Добрый день! Теоретически можно использовать QTcpSocket, у него есть метод connectToHost. Возможно, что проверка доступности через этот класс будет осуществляться несколько быстрее,...

QGraphicsScene

спасибо, за подробное объяснение строчки, а с зумом я разобрался, все работает

  • EVILEG
  • 15 января 2018 г. 17:21

Qt webgl

Насчёт проверки подключения клиента я не в курсе. Что касается экземпляров приложения, то из того, что я читал получается, что нет необходимости в нескольких экземплярах для нескольких кл...

  • EVILEG
  • 15 января 2018 г. 11:39

Проблема добавления #DEFINE при сборке CMak'ом

А Вы не пробовали сделать предкомпилированные библиотеки boost под свою систему, а потом уже подключать собранные библиотеки Boost`а? Просто один только boost может собираться на пару гиг...