{"id":198780,"date":"2013-10-24T23:48:05","date_gmt":"2013-10-24T19:48:05","guid":{"rendered":"http:\/\/savepearlharbor.com\/?p=198780"},"modified":"-0001-11-30T00:00:00","modified_gmt":"-0001-11-29T21:00:00","slug":"","status":"publish","type":"post","link":"https:\/\/savepearlharbor.com\/?p=198780","title":{"rendered":"<span class=\"post_title\">Qt Meta System over Network. \u0427\u0430\u0441\u0442\u044c 2 \u2014 \u0441\u0438\u0433\u043d\u0430\u043b\u044b \u0438 \u0441\u043b\u043e\u0442\u044b<\/span>"},"content":{"rendered":"<div class=\"content html_format\">   \t<img decoding=\"async\" src=\"http:\/\/habr.habrastorage.org\/post_images\/231\/e13\/1b3\/231e131b3b76d10af2cc0bfa5c96b413.jpg\" align=\"left\"\/><br \/>  <a href=\"http:\/\/habrahabr.ru\/post\/198270\/\">\u0427\u0430\u0441\u0442\u044c 1 \u2014 \u0421\u0432\u043e\u0439\u0441\u0442\u0432\u0430<\/a><br \/>  \u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0430\u0435\u043c \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u0442\u044c\u0441\u044f \u0441 \u043c\u0435\u0442\u0430\u0441\u0438\u0441\u0442\u0435\u043c\u043e\u0439 Qt. \u0412 \u044d\u0442\u043e\u0442 \u0440\u0430\u0437 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0445 \u0441\u0438\u0433\u043d\u0430\u043b\u043e\u0432 \u0438 \u0441\u043b\u043e\u0442\u043e\u0432.<br \/>  <a name=\"habracut\"><\/a>  <\/p>\n<h4>1. \u0421\u0438\u0433\u043d\u0430\u043b\u044b<\/h4>\n<p>  \u0410 \u0442\u043e\u0447\u043d\u0435\u0435 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 \u043a \u043b\u044e\u0431\u044b\u043c \u0441\u0438\u0433\u043d\u0430\u043b\u0430\u043c. \u0422\u0443\u0442 \u0432\u0441\u0451 \u043e\u0447\u0435\u043d\u044c \u043f\u043e\u0445\u043e\u0436\u0435 \u043d\u0430 \u0440\u0430\u0431\u043e\u0442\u0443 \u0441\u043e \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438, \u0442\u043e\u043b\u044c\u043a\u043e \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432 \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u043e\u0442 0 \u0434\u043e 10. \u0426\u0435\u043b\u044c \u2014 \u043f\u043e\u043b\u0443\u0447\u0438\u0442\u044c \u043a\u043b\u0430\u0441\u0441 DynamicQObject, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0441\u043c\u043e\u0436\u0435\u0442 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0438\u0442\u044c\u0441\u044f \u043a \u043b\u044e\u0431\u043e\u043c\u0443 \u0441\u0438\u0433\u043d\u0430\u043b\u0443 \u0438 \u043f\u0440\u0438 \u0435\u0433\u043e \u0430\u043a\u0442\u0438\u0432\u0430\u0446\u0438\u0438, \u0432\u044b\u0437\u0432\u0430\u0442\u044c \u043d\u0435\u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043c\u0435\u0442\u043e\u0434 \u0441 \u0438\u043c\u0435\u043d\u0435\u043c \u0441\u0438\u0433\u043d\u0430\u043b\u0430 \u0438 \u043f\u0435\u0440\u0435\u0434\u0430\u043d\u043d\u044b\u043c\u0438 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438 \u0432 \u0432\u0438\u0434\u0435 QVariantList. \u0422\u0430\u043a\u0436\u0435 \u0431\u0443\u0434\u0443 \u0440\u0430\u0437\u0431\u0438\u0432\u0430\u0442\u044c \u043a\u043e\u0434 \u043d\u0430 \u043a\u0443\u0441\u043e\u0447\u043a\u0438 \u0441 \u043a\u043e\u043c\u043c\u0435\u043d\u0442\u0430\u0440\u0438\u044f\u043c\u0438, \u043f\u0440\u0438\u0441\u0442\u0443\u043f\u0438\u043c:  <\/p>\n<div class=\"spoiler\"><b class=\"spoiler_title\">dynamicqobject.h<\/b><\/p>\n<div class=\"spoiler_text\">\n<pre><code class=\"cpp\">class DynamicQObject : public QObject { public:     DynamicQObject(QObject *mapToObject,                    const char *signalCatchMethod,                    QObject *parent = 0);      bool addSlot(QObject *object,                  const char *signal,                  const QString &slotName);     bool removeSlot(const QString &name);      bool addSignal(const QString &name, QObject *object, const char *slot);     bool removeSignal(const QString &name);     bool activate(const QString &signalName, const QVariantList &args);      int qt_metacall(QMetaObject::Call call, int id, void **arguments); private:     \/\/ virtual slots     bool containsSlot(const QString &name);     QObject *m_mapTo;     const char *m_catchMethod;     typedef struct {         bool isEmpty; \/\/ true after removeSlot()         QObject *object;         int signalIdx;         QString name; \/\/ virtual slot name         QVector&lt;int&gt; parameterTypes;     } slot_t;     QVector&lt;slot_t&gt; m_slotList;      \/\/ virtual signals     typedef struct {         bool isEmpty; \/\/ \/\/ true after removeSignal()         QObject *reciever;         int slotIdx;         QString name;         QVector&lt;int&gt; parameterTypes;     } signal_t;     QVector&lt;signal_t&gt; m_signalList;     QHash&lt;QString, int&gt; m_signalHash;     void *m_parameters[11]; \/\/ max 10 parameters + ret value }; <\/code><\/pre>\n<p>  <\/div>\n<\/div>\n<p>  <b>addSlot<\/b> \u2014 \u0441\u043e\u0437\u0434\u0430\u0451\u0442 \u043d\u043e\u0432\u044b\u0439 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u043b\u043e\u0442 \u0441 \u0438\u043c\u0435\u043d\u0435\u043c slotName \u0438 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0430\u0435\u0442 \u0435\u0433\u043e \u043a \u0441\u0438\u0433\u043d\u0430\u043b\u0443 signal \u043e\u0431\u044a\u0435\u043a\u0442\u0430 object. \u041f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u043a\u0430\u043a \u0442\u0443\u0442 \u0432\u0441\u0451 \u0443\u0441\u0442\u0440\u043e\u0435\u043d\u043e:  <\/p>\n<pre><code class=\"cpp\">bool DynamicQObject::addSlot(QObject *object,                              const char *signal,                              const QString &slotName) {     if (containsSlot(slotName))         return false;     if (signal[0] != '2') {         qWarning() &lt;&lt; &quot;Use SIGNAL() macro&quot;;         return false;     } <\/code><\/pre>\n<p>  \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c \u043d\u0435\u0442 \u043b\u0438 \u0443\u0436\u0435 \u0441\u043b\u043e\u0442\u0430 \u0441 \u0442\u0430\u043a\u0438\u043c \u0436\u0435 \u0438\u043c\u0435\u043d\u0435\u043c, \u0430 \u0442\u0430\u043a\u0436\u0435 \u043d\u0430\u043b\u0438\u0447\u0438\u0435 \u0441\u0438\u043c\u0432\u043e\u043b\u0430 &#8216;2&#8217; \u0432 \u043d\u0430\u0447\u0430\u043b\u0435 \u0435\u0433\u043e \u0438\u043c\u0435\u043d\u0438, \u044d\u0442\u043e\u0442 \u0441\u0438\u043c\u0432\u043e\u043b \u0434\u043e\u0431\u0430\u0432\u043b\u044f\u0435\u0442 \u043c\u0430\u043a\u0440\u043e\u0441 SIGNAL().  <\/p>\n<pre><code class=\"cpp\">    QByteArray theSignal = QMetaObject::normalizedSignature(&signal[1]);     int signalId = object-&gt;metaObject()-&gt;indexOfSignal(theSignal);     if (signalId &lt; 0) {         qWarning() &lt;&lt; &quot;signal&quot; &lt;&lt; signal &lt;&lt; &quot;doesn't exist&quot;;         return false;     }      QVector&lt;int&gt; parameterTypes;     QMetaMethod signalMethod = object-&gt;metaObject()-&gt;method(signalId);     for (int i = 0; i &lt; signalMethod.parameterCount(); ++i)         parameterTypes.push_back(signalMethod.parameterType(i)); <\/code><\/pre>\n<p>  \u0422\u0430\u043a\u0436\u0435 \u043a\u0430\u043a \u0438 \u0432 \u043f\u0440\u043e\u0448\u043b\u044b\u0439 \u0440\u0430\u0437, \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0438\u043d\u0434\u0435\u043a\u0441 \u0441\u0438\u0433\u043d\u0430\u043b\u0430, \u0430 \u0434\u0430\u043b\u0435\u0435 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u043f\u043e \u044d\u0442\u043e\u043c\u0443 \u0438\u043d\u0434\u0435\u043a\u0441\u0443 QMetaMethod. \u0418\u0437 \u043a\u043e\u0442\u043e\u0440\u043e\u0433\u043e \u043c\u044b \u043f\u043e \u043e\u0434\u043d\u043e\u043c\u0443 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u043c \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u044b \u0441\u0438\u0433\u043d\u0430\u043b\u0430, \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0438\u0437 \u043d\u0438\u0445 \u0432\u0435\u043a\u0442\u043e\u0440. \u0414\u0430\u043b\u0435\u0435 \u043c\u044b \u0431\u0443\u0434\u0435\u043c \u043f\u0440\u0438\u0432\u043e\u0434\u0438\u0442\u044c \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043d\u044b\u0435 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u044b \u0438\u043c\u0435\u043d\u043d\u043e \u043a \u044d\u0442\u0438\u043c \u0442\u0438\u043f\u0430\u043c.  <\/p>\n<pre><code class=\"cpp\">    int slotIdx = -1;     for (int i = 0; i &lt; m_slotList.count(); ++i) {         if (m_slotList[i].isEmpty == true) {             slotIdx = i;             break;         }     }     bool addEntry = false;     if (slotIdx == -1) {         addEntry = true;         slotIdx = m_slotList.count();     } <\/code><\/pre>\n<p>  \u0417\u0434\u0435\u0441\u044c \u0432\u0441\u0451 \u0441\u043e\u0432\u0441\u0435\u043c \u043f\u0440\u043e\u0441\u0442\u043e, \u043d\u0430\u0445\u043e\u0434\u0438\u043c \u043f\u0443\u0441\u0442\u0443\u044e \u0437\u0430\u043f\u0438\u0441\u044c (\u043b\u0438\u0431\u043e \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u043d\u043e\u0432\u0443\u044e) \u0432 m_slotList, \u043a\u0443\u0434\u0430 \u043c\u044b \u0441\u043e\u0445\u0440\u0430\u043d\u0438\u043c \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \u043e \u0441\u043e\u0437\u0434\u0430\u043d\u043d\u043e\u043c \u0441\u043b\u043e\u0442\u0435. \u041f\u0443\u0441\u0442\u044b\u0435 \u0437\u0430\u043f\u0438\u0441\u0438 \u043e\u0431\u0440\u0430\u0437\u0443\u044e\u0442\u0441\u044f \u043f\u043e\u0441\u043b\u0435 \u0443\u0434\u0430\u043b\u0435\u043d\u0438\u044f \u0441\u043b\u043e\u0442\u043e\u0432 ( removeSlot() ), \u0442.\u043a. \u0438\u043d\u0434\u0435\u043a\u0441\u044b \u0443 \u043d\u0430\u0441 \u043f\u0440\u0438\u0432\u044f\u0437\u0430\u043d\u044b \u043a \u0441\u043b\u043e\u0442\u0430\u043c, \u0438\u0445 \u0441\u0434\u0432\u0438\u0433 \u043f\u0440\u0438\u0432\u0451\u043b \u0431\u044b \u043a \u0432\u044b\u0437\u043e\u0432\u0443 \u043d\u0435\u0432\u0435\u0440\u043d\u044b\u0445 \u0441\u043b\u043e\u0442\u043e\u0432. \u041c\u043e\u0436\u043d\u043e \u0431\u044b\u043b\u043e \u0431\u044b \u043f\u0440\u0438\u043c\u0435\u043d\u0438\u0442\u044c \u0437\u0434\u0435\u0441\u044c QHash \u0438\u043b\u0438 QMap, \u043d\u043e \u044f \u043f\u043e\u0441\u0447\u0438\u0442\u0430\u043b, \u0447\u0442\u043e \u0443\u0434\u0430\u043b\u044f\u044e\u0442 \u0441\u043b\u043e\u0442\u044b \u0433\u043e\u0440\u0430\u0437\u0434\u043e \u0440\u0435\u0436\u0435, \u0447\u0435\u043c \u0441\u043e\u0437\u0434\u0430\u044e\u0442, \u0430 \u0432\u044b\u0437\u044b\u0432\u0430\u044e\u0442 \u043e\u0447\u0435\u043d\u044c \u0447\u0430\u0441\u0442\u043e, \u0442\u0430\u043a \u0447\u0442\u043e \u0432\u0435\u043a\u0442\u043e\u0440 \u044f\u0432\u043d\u043e \u044d\u0444\u0444\u0435\u043a\u0442\u0438\u0432\u043d\u0435\u0435, \u0442.\u043a. \u0443 \u043d\u0435\u0433\u043e \u0434\u043e\u0441\u0442\u0443\u043f \u043f\u043e \u0438\u043d\u0434\u0435\u043a\u0441\u0443 \u0432\u044b\u043f\u043e\u043b\u043d\u044f\u0435\u0442\u0441\u044f \u0437\u0430 O(1), \u0430 \u0443 QMap \u0438 QHash \u0432 \u0445\u0443\u0434\u0448\u0435\u043c \u0441\u043b\u0443\u0447\u0430\u0435 \u0437\u0430 O(logn) \u0438 O(n) \u0441\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e.<br \/>  \u0421\u043e\u0431\u0441\u0442\u0432\u0435\u043d\u043d\u043e \u043e\u0441\u0442\u0430\u043b\u043e\u0441\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0438\u0442\u044c\u0441\u044f \u043a \u0441\u0438\u0433\u043d\u0430\u043b\u0443:  <\/p>\n<pre><code class=\"cpp\">    if (!QMetaObject::connect(object, signalId,                                 this, slotIdx + metaObject()-&gt;methodCount())) {         qWarning() &lt;&lt; &quot;can't connect&quot; &lt;&lt; signal &lt;&lt; &quot;signal to virtual slot&quot;;         return false;     }      if (addEntry) {         m_slotList.push_back({false, object, signalId, slotName, parameterTypes});     } else {         slot_t &slot = m_slotList[slotIdx];         slot.isEmpty = false;         slot.object = object;         slot.signalIdx = signalId;         slot.name = slotName;         slot.parameterTypes = parameterTypes;     }      return true; } <\/code><\/pre>\n<p>  \u0418 \u0441\u043e\u0445\u0440\u0430\u043d\u0438\u0442\u044c \u0432\u0441\u044e \u043d\u0435\u043e\u0431\u0445\u043e\u0434\u0438\u043c\u0443\u044e \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \u043e \u043d\u0451\u043c.<\/p>\n<h5>\u0412\u044b\u0437\u043e\u0432 \u0441\u043b\u043e\u0442\u0430<\/h5>\n<p>  \u0422\u0443\u0442 \u0432\u0441\u0451 \u043f\u043e\u0445\u043e\u0436\u0435 \u043d\u0430 \u0444\u043e\u043a\u0443\u0441 \u0441\u043e \u0441\u0432\u043e\u0439\u0441\u0442\u0432\u0430\u043c\u0438, \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0441\u0432\u043e\u044e \u0440\u0435\u0430\u043b\u0438\u0437\u0430\u0446\u0438\u044e qt_metacall, \u0442\u043e\u043b\u044c\u043a\u043e \u0443 \u043d\u0430\u0441 \u0442\u0435\u043f\u0435\u0440\u044c \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432 \u043c\u043e\u0436\u0435\u0442 \u0431\u044b\u0442\u044c \u0431\u043e\u043b\u044c\u0448\u0435:  <\/p>\n<pre><code class=\"cpp\">int DynamicQObject::qt_metacall(QMetaObject::Call call, int id, void **arguments) {     id = QObject::qt_metacall(call, id, arguments);     if (id &lt; 0 || call != QMetaObject::InvokeMetaMethod)         return id;     Q_ASSERT(id &lt; m_slotList.size()); <\/code><\/pre>\n<p>  \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0432\u044b\u0437\u044b\u0432\u0430\u044e\u0442 \u0434\u0435\u0439\u0441\u0442\u0432\u0438\u0442\u0435\u043b\u044c\u043d\u043e \u0441\u043b\u043e\u0442 (\u043c\u0435\u0442\u0430\u043c\u0435\u0442\u043e\u0434), \u0430 \u0442\u0430\u043a\u0436\u0435 \u0447\u0442\u043e \u0438\u043d\u0434\u0435\u043a\u0441 \u043a\u043e\u0440\u0440\u0435\u043a\u0442\u043d\u044b\u0439.  <\/p>\n<pre><code class=\"cpp\">    const slot_t &slotInfo = m_slotList[id];     QVariantList parameters;     for (int i = 0; i &lt; slotInfo.parameterTypes.count(); ++i) {         void *parameter = arguments[i + 1];         parameters.append(QVariant(slotInfo.parameterTypes[i], parameter));     } <\/code><\/pre>\n<p>  \u0414\u043e\u0441\u0442\u0430\u0451\u043c \u0440\u0430\u043d\u0435\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u0451\u043d\u043d\u0443\u044e \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e \u043e \u0441\u043b\u043e\u0442\u0435. \u0418 \u0434\u0430\u043b\u044c\u0448\u0435 \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0441\u043f\u0438\u0441\u043e\u043a QVariant-\u043e\u0432 \u0438\u0437 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u043d\u044b\u0439 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432 \u0438 \u0441\u043e\u0445\u0440\u0430\u043d\u0451\u043d\u043d\u044b\u0445 \u0442\u0438\u043f\u043e\u0432.  <\/p>\n<pre><code class=\"cpp\">    QMetaObject::invokeMethod(m_mapTo, m_catchMethod,                               Q_ARG(QString, slotInfo.name),                               Q_ARG(QVariantList, parameters));     return -1; } <\/code><\/pre>\n<p>  \u041e\u0441\u0442\u0430\u043b\u043e\u0441\u044c \u0442\u043e\u043b\u044c\u043a\u043e \u0432\u044b\u0437\u0432\u0430\u0442\u044c \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u044b\u0439 \u0432 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440\u0435 \u043c\u0435\u0442\u043e\u0434 \u0441 \u0438\u043c\u0435\u043d\u0435\u043c \u0441\u043b\u043e\u0442\u0430, \u0438 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438.<\/p>\n<p>  \u0412 removeSlot() \u043d\u0435\u0442 \u043d\u0438\u0447\u0435\u0433\u043e \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0433\u043e, \u0442\u0430\u043a \u0447\u0442\u043e \u043f\u043e\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0441\u0440\u0430\u0437\u0443 \u043d\u0430 \u043f\u0440\u0438\u043c\u0435\u0440 \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u043d\u0438\u044f:  <\/p>\n<pre><code class=\"cpp\">Reciever reciever; DynamicQObject dynamic(&reciever, &quot;signalCatched&quot;);  Tester tester; dynamic.addSlot(&tester, SIGNAL(signal1(int,int,QString)), &quot;myslot1&quot;); dynamic.addSlot(&tester, SIGNAL(signal2(QPoint)), &quot;myslot2&quot;);  tester.emitSignal1(); tester.emitSingal2(); <\/code><\/pre>\n<p>  \u0412 \u043a\u043e\u043d\u0441\u0442\u0440\u0443\u043a\u0442\u043e\u0440 DynamicQObject \u043f\u0435\u0440\u0435\u0434\u0430\u0435\u043c \u0443\u043a\u0430\u0437\u0430\u0442\u0435\u043b\u044c \u043d\u0430 \u043b\u044e\u0431\u043e\u0433\u043e \u043d\u0430\u0441\u043b\u0435\u0434\u043d\u0438\u043a\u0430 QObject \u0438 \u0438\u043c\u044f \u043c\u0435\u0442\u043e\u0434\u0430 (Q_INVOKABLE), \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u0431\u0443\u0434\u0435\u0442 \u0432\u044b\u0437\u044b\u0432\u0430\u0442\u044c \u043f\u0440\u0438 \u043f\u043e\u043b\u0443\u0447\u0435\u043d\u0438\u0438 \u043b\u044e\u0431\u043e\u0433\u043e \u0441\u0438\u0433\u043d\u0430\u043b\u0430. <br \/>  \u0412 \u043a\u043b\u0430\u0441\u0441\u0435 Reciever \u0443 \u043d\u0430\u0441 \u0435\u0441\u0442\u044c \u0434\u043b\u044f \u0438\u043c\u0435\u0435\u0442\u0441\u044f \u0435\u0441\u0442\u044c \u0441\u043b\u0435\u0434\u0443\u044e\u0449\u0438\u0439 \u043c\u0435\u0442\u043e\u0434: <i>Q_INVOKABLE void signalCatched(const QString &#038;signalName, const QVariantList &#038;args)<\/i>, \u043a\u043e\u0442\u043e\u0440\u044b\u0439 \u043f\u0440\u043e\u0441\u0442\u043e \u0432\u044b\u0432\u043e\u0434\u0438\u0442\u044c \u0432 \u043a\u043e\u043d\u0441\u043e\u043b\u044c \u0441\u0432\u043e\u0438 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u044b.<br \/>  \u0410 \u0432 Tester \u0435\u0441\u0442\u044c \u0434\u0432\u0430 \u0441\u0438\u0433\u043d\u0430\u043b\u0430 \u0441 \u0443\u043a\u0430\u0437\u0430\u043d\u043d\u044b\u043c\u0438 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0430\u043c\u0438, \u0438 \u0434\u0432\u0435 \u0444\u0443\u043d\u043a\u0446\u0438\u0438 \u0433\u0435\u043d\u0435\u0440\u0438\u0440\u0443\u044e\u0449\u0438\u0435 \u0438\u0445, \u0434\u0443\u043c\u0430\u044e \u0432\u0441\u0451 \u0434\u043e\u043b\u0436\u043d\u043e \u0431\u044b\u0442\u044c \u043f\u043e\u043d\u044f\u0442\u043d\u043e. \u0417\u0430\u043f\u0443\u0441\u043a\u0430\u0435\u043c:  <\/p>\n<blockquote><p>\u00abmyslot1\u00bb (QVariant(int, 123), QVariant(int, 456), QVariant(QString, \u00abstr\u00bb) ) <br \/>  \u00abmyslot2\u00bb (QVariant(QPoint, QPoint(3,4) ) ) <\/p><\/blockquote>\n<p>  \u0412\u0438\u0434\u0438\u043c, \u0447\u0442\u043e \u0432\u0441\u0451 \u0440\u0430\u0431\u043e\u0442\u0430\u0435\u0442:) \u041c\u043e\u0436\u043d\u043e \u0438\u0441\u043f\u043e\u043b\u044c\u0437\u043e\u0432\u0430\u0442\u044c \u0430\u0431\u0441\u043e\u043b\u044e\u0442\u043d\u043e \u043b\u044e\u0431\u044b\u0435 \u0442\u0438\u043f\u044b \u0438\u0437\u0432\u0435\u0441\u0442\u043d\u044b\u0435 \u043c\u0435\u0442\u0430\u0441\u0438\u0441\u0442\u0435\u043c\u0435 Qt.<\/p>\n<h4>2. \u0421\u043b\u043e\u0442\u044b<\/h4>\n<p>  \u0422.\u0435. \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0445 \u0441\u0438\u0433\u043d\u0430\u043b\u043e\u0432 \u0438 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0435\u043d\u0438\u0435 \u0438\u0445 \u043a \u043e\u0431\u044b\u0447\u043d\u044b\u043c \u0441\u043b\u043e\u0442\u0430\u043c, \u0442\u0443\u0442 \u043b\u0435\u0433\u043a\u043e \u043f\u0435\u0440\u0435\u043f\u0443\u0442\u0430\u0442\u044c \u043e\u0434\u043d\u043e \u0441 \u0434\u0440\u0443\u0433\u0438\u043c\u2026 \u0417\u0430 \u044d\u0442\u043e \u043e\u0442\u0432\u0435\u0447\u0430\u044e\u0442 3 \u0444\u0443\u043d\u043a\u0446\u0438\u0438: addSignal(), removeSignal() \u0438 activate().<br \/>  \u0420\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0441\u0430\u043c\u043e\u0435 \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0435. \u0421\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0441\u0438\u0433\u043d\u0430\u043b\u0430:  <\/p>\n<pre><code class=\"cpp\">bool DynamicQObject::addSignal(const QString &name, QObject *object, const char *slot) {     if (slot[0] != '1') {         qWarning() &lt;&lt; &quot;Use SLOT() macro&quot;;         return false;     }     int slotIdx = object-&gt;metaObject()-&gt;             indexOfSlot(&slot[1]); \/\/ without 1 added by SLOT() macro     if (slotIdx &lt; 0) {         qWarning() &lt;&lt; slot &lt;&lt; &quot;slot didn't exist&quot;;         return false;     } <\/code><\/pre>\n<p>  \u041a\u0430\u043a \u043e\u0431\u044b\u0447\u043d\u043e, \u043f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0432\u0441\u0451 \u0438\u0434\u0451\u0442 \u043a\u0430\u043a \u043d\u0430\u0434\u043e.  <\/p>\n<pre><code class=\"cpp\">    QVector&lt;int&gt; parameterTypes;     QMetaMethod slotMethod = object-&gt;metaObject()-&gt;method(slotIdx);     for (int i = 0; i &lt; slotMethod.parameterCount(); ++i)         parameterTypes.push_back(slotMethod.parameterType(i));      int signalIdx = -1;     for (int i = 0; i &lt; m_slotList.count(); ++i) {         if (m_slotList[i].isEmpty == true) {             signalIdx = i;             break;         }     }     bool addEntry = false;     if (signalIdx == -1) {         addEntry = true;         signalIdx = m_signalList.count();     } <\/code><\/pre>\n<p>  \u0410\u043d\u0430\u043b\u043e\u0433\u0438\u0447\u043d\u043e \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0432\u0435\u043a\u0442\u043e\u0440 \u0442\u0438\u043f\u043e\u0432 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432.  <\/p>\n<pre><code class=\"cpp\">    if (!QMetaObject::connect(this, signalIdx + metaObject()-&gt;methodCount(),                               object, slotIdx)) {         qWarning() &lt;&lt; &quot;can't connect virtual signal&quot; &lt;&lt; name &lt;&lt; &quot;to slot&quot; &lt;&lt; slot;         return false;     }      if (addEntry) {         m_signalList.append({false, object, slotIdx, name, parameterTypes});     } else {         signal_t &signal = m_signalList[signalIdx];         signal.isEmpty = false;         signal.reciever = object;         signal.slotIdx = slotIdx;         signal.name = name;         signal.parameterTypes = parameterTypes;     }     m_signalHash.insert(name, signalIdx);      return true; } <\/code><\/pre>\n<p>  \u0418 \u0441\u043d\u043e\u0432\u0430 \u0432\u0441\u0451 \u043f\u043e\u0447\u0442\u0438 \u0442\u0430\u043a\u0436\u0435 \u2014 \u043f\u043e\u0434\u043a\u043b\u044e\u0447\u0430\u0435\u043c \u043d\u0430\u0448 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u0438\u0433\u043d\u0430\u043b \u043a \u0441\u043b\u043e\u0442\u0443, \u0430 \u0442\u0430\u043a\u0436\u0435 \u0441\u043e\u0445\u0440\u0430\u043d\u044f\u0435\u043c \u0432 m_signalHash \u043f\u0430\u0440\u0443 <i>\u0438\u043c\u044f \u2014 \u0438\u043d\u0434\u0435\u043a\u0441 \u0441\u0438\u0433\u043d\u0430\u043b\u0430<\/i>. \u0422\u0430\u043a\u0438\u043c \u043e\u0431\u0440\u0430\u0437\u043e\u043c \u043f\u0440\u0438 \u0430\u043a\u0442\u0438\u0432\u0430\u0446\u0438\u0438 \u0441\u0438\u0433\u043d\u0430\u043b\u0430, \u043c\u044b \u043f\u043e\u043b\u0443\u0447\u0438\u043c \u0435\u0433\u043e \u0438\u043d\u0434\u0435\u043a\u0441 \u0438\u0437 \u0438\u043c\u0435\u043d\u0438.<br \/>  \u0423\u0434\u0430\u043b\u0435\u043d\u0438\u0435 \u0440\u0430\u0441\u0441\u043c\u0430\u0442\u0440\u0438\u0432\u0430\u0442\u044c \u043d\u0435 \u0431\u0443\u0434\u0435\u043c, \u0430 \u0442\u043e \u0438 \u0442\u0430\u043a \u043e\u0447\u0435\u043d\u044c \u043c\u043d\u043e\u0433\u043e \u043a\u043e\u0434\u0430\u2026 <\/p>\n<h5>\u0410\u043a\u0442\u0438\u0432\u0430\u0446\u0438\u044f \u0441\u0438\u0433\u043d\u0430\u043b\u0430<\/h5>\n<p>  \u0422\u0443\u0442 \u0443\u0436\u0435 \u0435\u0441\u0442\u044c \u043d\u043e\u0432\u044b\u0435 \u043c\u043e\u043c\u0435\u043d\u0442\u044b, \u0430 \u0438\u043c\u0435\u043d\u043d\u043e \u043f\u0440\u0438\u0432\u0435\u0434\u0435\u043d\u0438\u0435 \u0442\u0438\u043f\u043e\u0432.  <\/p>\n<pre><code class=\"cpp\">bool DynamicQObject::activate(const QString &signalName, const QVariantList &args) {     int signalIdx = m_signalHash.value(signalName, -1);     if (signalIdx == -1) {         qWarning() &lt;&lt; &quot;signal&quot; &lt;&lt; signalName &lt;&lt; &quot;doesn't exist&quot;;         return false;     }      signal_t &signal = m_signalList[signalIdx]; <\/code><\/pre>\n<p>  \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0441\u0438\u0433\u043d\u0430\u043b \u0441\u0443\u0449\u0435\u0441\u0442\u0432\u0443\u0435\u0442 \u0438 \u0438\u0437\u0432\u043b\u0435\u043a\u0430\u0435\u043c \u0441\u043e\u0445\u0440\u0430\u043d\u0451\u043d\u043d\u0443\u044e \u0440\u0430\u043d\u0435\u0435 \u0438\u043d\u0444\u043e\u0440\u043c\u0430\u0446\u0438\u044e.  <\/p>\n<pre><code class=\"cpp\">    if (args.count() &lt; signal.parameterTypes.count()) {         qWarning() &lt;&lt; &quot;parameters count mismatch:&quot; &lt;&lt; signalName                    &lt;&lt; &quot;provided:&quot; &lt;&lt; args.count()                    &lt;&lt; &quot;need &gt;=:&quot; &lt;&lt; signal.parameterTypes.count();         return false;     } <\/code><\/pre>\n<p>  \u041f\u0440\u043e\u0432\u0435\u0440\u044f\u0435\u043c, \u0447\u0442\u043e \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432 \u0434\u043e\u0441\u0442\u0430\u0442\u043e\u0447\u043d\u043e, \u043b\u0438\u0448\u043d\u0438\u0435 \u043f\u0440\u043e\u0441\u0442\u043e \u043e\u0442\u0431\u0440\u043e\u0441\u0438\u043c\u2026  <\/p>\n<pre><code class=\"cpp\">    QVariantList argsCopy = args;      for (int i = 0; i &lt; signal.parameterTypes.count(); ++i) {         if (!argsCopy[i].convert(signal.parameterTypes[i])) {             qWarning() &lt;&lt; &quot;can't cast parameter&quot; &lt;&lt; i &lt;&lt; signalName;             return false;         }         m_parameters[i + 1] = argsCopy[i].data();     } <\/code><\/pre>\n<p>  \u0421\u043e\u0437\u0434\u0430\u0451\u043c \u043a\u043e\u043f\u0438\u044e \u0441\u043f\u0438\u0441\u043a\u0430 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432, \u0442.\u043a. \u043e\u043d \u043a\u043e\u043d\u0441\u0442\u0430\u043d\u0442\u043d\u044b\u0439, \u0430 \u043d\u0430\u043c \u043d\u0443\u0436\u043d\u043e \u043f\u0440\u0438\u0432\u043e\u0434\u0438\u0442\u044c \u0442\u0438\u043f\u044b, \u0442.\u0435. \u043c\u0435\u043d\u044f\u0442\u044c \u0435\u0433\u043e. \u0410 \u0434\u0430\u043b\u0435\u0435 \u043f\u0440\u043e\u0431\u0443\u0435\u043c \u043f\u0440\u0438\u0432\u0435\u0441\u0442\u0438 \u043a\u0430\u0436\u0434\u044b\u0439 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442 \u043a \u043d\u0443\u0436\u043d\u043e\u043c\u0443 \u0442\u0438\u043f\u0443. \u0422\u0430\u043a, \u043d\u0430\u043f\u0440\u0438\u043c\u0435\u0440, \u043c\u043e\u0436\u043d\u043e \u0432\u044b\u0437\u0432\u0430\u0442\u044c \u0441\u043b\u043e\u0442 \u0441 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u043c \u0442\u0438\u043f\u0430 int, \u043f\u0435\u0440\u0435\u0434\u0430\u0432 activate \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442 \u0442\u0438\u043f\u0430 QString, \u0433\u043b\u0430\u0432\u043d\u043e\u0435 \u0447\u0442\u043e\u0431\u044b \u0441\u0442\u0440\u043e\u043a\u0430 \u0441\u043e\u0434\u0435\u0440\u0436\u0430\u043b\u0430 \u0447\u0438\u0441\u043b\u043e.<br \/>  \u041f\u0440\u0438\u043c\u0435\u0440:  <\/p>\n<pre><code class=\"cpp\">    DynamicQObject dynamic(&reciever, &quot;signalCatched&quot;);      Reciever reciever;     dynamic.addSignal(&quot;virtual_signal&quot;, &reciever, SLOT(slot1(int,QString)));      dynamic.activate(&quot;virtual_signal&quot;, QVariantList() &lt;&lt; &quot;123&quot; &lt;&lt; QString(&quot;qwerty&quot;) &lt;&lt; 2 &lt;&lt; 3); <\/code><\/pre>\n<p>  \u0421\u043e\u043e\u0442\u0432\u0435\u0442\u0441\u0442\u0432\u0435\u043d\u043d\u043e Reciever \u0441\u043e\u0434\u0435\u0440\u0436\u0438\u0442 \u043e\u0431\u044b\u0447\u043d\u044b\u0439 \u0441\u043b\u043e\u0442. \u041c\u044b \u0441\u043e\u0437\u0434\u0430\u0451\u043c \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0439 \u0441\u0438\u0433\u043d\u0430\u043b \u0438 \u0432\u044b\u0437\u044b\u0432\u0430\u0435\u043c \u0435\u0433\u043e \u0441\u043e \u0441\u043f\u0438\u0441\u043a\u043e\u043c \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u043e\u0432.<br \/>  \u0412\u043e\u0442 \u0447\u0442\u043e \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f:  <\/p>\n<blockquote><p>slot1_call 123 \u00abqwerty\u00bb<\/p><\/blockquote>\n<p>  \u0414\u0432\u0430 \u043b\u0438\u0448\u043d\u0438\u0445 \u0430\u0440\u0433\u0443\u043c\u0435\u043d\u0442\u0430 \u043e\u0442\u0431\u0440\u043e\u0441\u0438\u043b\u0438\u0441\u044c, \u0430 \u0434\u043b\u044f \u043f\u0435\u0440\u0432\u043e\u0433\u043e \u0431\u044b\u043b\u043e \u0432\u044b\u043f\u043e\u043b\u043d\u0435\u043d\u043e \u043f\u0440\u0435\u043e\u0431\u0440\u0430\u0437\u043e\u0432\u0430\u043d\u0438\u0435 \u0442\u0438\u043f\u043e\u0432, \u043d\u0435 \u0443\u0432\u0435\u0440\u0435\u043d, \u0447\u0442\u043e \u044d\u0442\u043e \u043e\u0447\u0435\u043d\u044c \u043d\u0443\u0436\u043d\u0430\u044f \u0432\u0435\u0449\u044c, \u043d\u043e \u0442\u0443\u0442 \u043e\u043d\u0430 \u043f\u043e\u043b\u0443\u0447\u0430\u0435\u0442\u0441\u044f \u0441\u0430\u043c\u0430 \u043f\u043e \u0441\u0435\u0431\u0435, \u0445\u043e\u0442\u044f \u043c\u043e\u0436\u043d\u043e \u0438 \u0437\u0430\u043f\u0440\u0435\u0442\u0438\u0442\u044c \u043f\u043e\u0434\u043e\u0431\u043d\u043e\u0435 \u043f\u043e\u0432\u0435\u0434\u0435\u043d\u0438\u0435\u2026<\/p>\n<p>  \u041f\u043e\u043a\u0430 \u0432\u0441\u0451. \u0414\u0430\u043b\u0435\u0435 \u0440\u0430\u0437\u0431\u0435\u0440\u0451\u043c\u0441\u044f \u0441 \u043c\u0435\u0442\u043e\u0434\u0430\u043c\u0438, \u0438 \u043f\u0440\u0438\u043a\u0440\u0443\u0442\u0438\u043c \u043f\u0440\u043e\u0441\u0442\u0435\u043d\u044c\u043a\u0438\u0439 \u0441\u0435\u0442\u0435\u0432\u043e\u0439 \u0438\u043d\u0442\u0435\u0440\u0444\u0435\u0439\u0441\u2026<\/p>\n<p>  P.S. \u0412 \u043d\u0430\u043f\u0438\u0441\u0430\u043d\u0438\u0438 \u0434\u0430\u043d\u043d\u043e\u0433\u043e \u043a\u043b\u0430\u0441\u0441\u0430, \u043e\u0447\u0435\u043d\u044c \u043f\u043e\u043c\u043e\u0433\u043b\u0430 \u0441\u0442\u0430\u0442\u044c\u044f: <a href=\"http:\/\/doc.qt.digia.com\/qq\/qq16-dynamicqobject.html\">Dynamic Signals and Slots<\/a>, \u0438\u0437 \u043a\u043e\u0442\u043e\u0440\u043e\u0439 \u043c\u043e\u0436\u043d\u043e \u0443\u0437\u043d\u0430\u0442\u044c \u043c\u043d\u043e\u0433\u043e \u0432\u0441\u0435\u0433\u043e \u0438\u043d\u0442\u0435\u0440\u0435\u0441\u043d\u043e\u0433\u043e.    \t<\/p>\n<div class=\"clear\"><\/div>\n<\/p><\/div>\n<p> \u0441\u0441\u044b\u043b\u043a\u0430 \u043d\u0430 \u043e\u0440\u0438\u0433\u0438\u043d\u0430\u043b \u0441\u0442\u0430\u0442\u044c\u0438 <a href=\"http:\/\/habrahabr.ru\/post\/198780\/\"> http:\/\/habrahabr.ru\/post\/198780\/<\/a><\/p>\n","protected":false},"excerpt":{"rendered":"<div class=\"content html_format\">   \t<img decoding=\"async\" src=\"http:\/\/habr.habrastorage.org\/post_images\/231\/e13\/1b3\/231e131b3b76d10af2cc0bfa5c96b413.jpg\" align=\"left\"\/><br \/>  <a href=\"http:\/\/habrahabr.ru\/post\/198270\/\">\u0427\u0430\u0441\u0442\u044c 1 \u2014 \u0421\u0432\u043e\u0439\u0441\u0442\u0432\u0430<\/a><br \/>  \u041f\u0440\u043e\u0434\u043e\u043b\u0436\u0430\u0435\u043c \u0440\u0430\u0437\u0431\u0438\u0440\u0430\u0442\u044c\u0441\u044f \u0441 \u043c\u0435\u0442\u0430\u0441\u0438\u0441\u0442\u0435\u043c\u043e\u0439 Qt. \u0412 \u044d\u0442\u043e\u0442 \u0440\u0430\u0437 \u0440\u0430\u0441\u0441\u043c\u043e\u0442\u0440\u0438\u043c \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u0435 \u0432\u0438\u0440\u0442\u0443\u0430\u043b\u044c\u043d\u044b\u0445 \u0441\u0438\u0433\u043d\u0430\u043b\u043e\u0432 \u0438 \u0441\u043b\u043e\u0442\u043e\u0432.  <\/p>\n","protected":false},"author":1,"featured_media":0,"comment_status":"open","ping_status":"open","sticky":false,"template":"","format":"standard","meta":{"footnotes":""},"categories":[],"tags":[],"class_list":["post-198780","post","type-post","status-publish","format-standard","hentry"],"_links":{"self":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/198780","targetHints":{"allow":["GET"]}}],"collection":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts"}],"about":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/types\/post"}],"author":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/users\/1"}],"replies":[{"embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcomments&post=198780"}],"version-history":[{"count":0,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=\/wp\/v2\/posts\/198780\/revisions"}],"wp:attachment":[{"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fmedia&parent=198780"}],"wp:term":[{"taxonomy":"category","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Fcategories&post=198780"},{"taxonomy":"post_tag","embeddable":true,"href":"https:\/\/savepearlharbor.com\/index.php?rest_route=%2Fwp%2Fv2%2Ftags&post=198780"}],"curies":[{"name":"wp","href":"https:\/\/api.w.org\/{rel}","templated":true}]}}