Refresh cache for method VDomDocument::elementById() in separate thread.

--HG--
branch : develop
This commit is contained in:
Roman Telezhynskyi 2018-01-26 14:43:52 +02:00
parent d5843ceb6d
commit 01acee890b
4 changed files with 74 additions and 43 deletions

View File

@ -239,13 +239,12 @@ void GatherTokens(QSet<QString> &tokens, const QList<QString> &tokenList)
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
VAbstractPattern::VAbstractPattern(QObject *parent) VAbstractPattern::VAbstractPattern(QObject *parent)
: QObject(parent), : VDomDocument(parent),
VDomDocument(), nameActivPP(),
nameActivPP(QString()),
cursor(0), cursor(0),
toolsOnRemove(QVector<VDataTool*>()), toolsOnRemove(QVector<VDataTool*>()),
history(QVector<VToolRecord>()), history(QVector<VToolRecord>()),
patternPieces(QStringList()), patternPieces(),
modified(false) modified(false)
{} {}

View File

@ -79,7 +79,7 @@ struct VFinalMeasurement
QT_WARNING_POP QT_WARNING_POP
class VAbstractPattern : public QObject, public VDomDocument class VAbstractPattern : public VDomDocument
{ {
Q_OBJECT Q_OBJECT
public: public:

View File

@ -59,6 +59,9 @@
#include <QXmlSchemaValidator> #include <QXmlSchemaValidator>
#include <QtDebug> #include <QtDebug>
#include <QXmlStreamWriter> #include <QXmlStreamWriter>
#include <QTimer>
#include <QtConcurrentRun>
#include <QFutureWatcher>
namespace namespace
{ {
@ -190,10 +193,20 @@ const QString VDomDocument::TagUnit = QStringLiteral("unit");
const QString VDomDocument::TagLine = QStringLiteral("line"); const QString VDomDocument::TagLine = QStringLiteral("line");
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
VDomDocument::VDomDocument() VDomDocument::VDomDocument(QObject *parent)
: QDomDocument(), : QObject(parent),
map() QDomDocument(),
{} m_elementIdCache(),
m_refreshCacheTimer(new QTimer(this)),
m_watcher(new QFutureWatcher<QHash<quint32, QDomElement>>(this))
{
m_refreshCacheTimer->setTimerType(Qt::VeryCoarseTimer);
m_refreshCacheTimer->setInterval(10000);
m_refreshCacheTimer->setSingleShot(true);
connect(m_refreshCacheTimer, &QTimer::timeout, this, &VDomDocument::RefreshElementIdCache);
connect(m_watcher, &QFutureWatcher<QHash<quint32, QDomElement>>::finished, this, &VDomDocument::CacheRefreshed);
m_refreshCacheTimer->start();
}
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
QDomElement VDomDocument::elementById(quint32 id, const QString &tagName) QDomElement VDomDocument::elementById(quint32 id, const QString &tagName)
@ -203,21 +216,21 @@ QDomElement VDomDocument::elementById(quint32 id, const QString &tagName)
return QDomElement(); return QDomElement();
} }
if (map.contains(id)) if (m_elementIdCache.contains(id))
{ {
const QDomElement e = map[id]; const QDomElement e = m_elementIdCache.value(id);
if (e.parentNode().nodeType() != QDomNode::BaseNode) if (e.parentNode().nodeType() != QDomNode::BaseNode)
{ {
return e; return e;
} }
map.remove(id); m_elementIdCache.remove(id);
} }
if (tagName.isEmpty()) if (tagName.isEmpty())
{ {
if (this->find(this->documentElement(), id)) if (VDomDocument::find(m_elementIdCache, this->documentElement(), id))
{ {
return map[id]; return m_elementIdCache.value(id);
} }
} }
else else
@ -228,19 +241,12 @@ QDomElement VDomDocument::elementById(quint32 id, const QString &tagName)
const QDomElement domElement = list.at(i).toElement(); const QDomElement domElement = list.at(i).toElement();
if (not domElement.isNull() && domElement.hasAttribute(AttrId)) if (not domElement.isNull() && domElement.hasAttribute(AttrId))
{ {
try const quint32 elementId = GetParametrUInt(domElement, AttrId, NULL_ID_STR);
{
const quint32 elementId = GetParametrUInt(domElement, AttrId, NULL_ID_STR);
this->map[elementId] = domElement; m_elementIdCache.insert(elementId, domElement);
if (elementId == id) if (elementId == id)
{
return domElement;
}
}
catch (const VExceptionConversionError &)
{ {
// do nothing return domElement;
} }
} }
} }
@ -252,27 +258,21 @@ QDomElement VDomDocument::elementById(quint32 id, const QString &tagName)
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
/** /**
* @brief Find element by id. * @brief Find element by id.
* @param cache cache with element ids
* @param node node * @param node node
* @param id id value * @param id id value
* @return true if found * @return true if found
*/ */
bool VDomDocument::find(const QDomElement &node, quint32 id) bool VDomDocument::find(QHash<quint32, QDomElement> &cache, const QDomElement &node, quint32 id)
{ {
if (node.hasAttribute(AttrId)) if (node.hasAttribute(AttrId))
{ {
try const quint32 elementId = GetParametrUInt(node, AttrId, NULL_ID_STR);
{
const quint32 elementId = GetParametrUInt(node, AttrId, NULL_ID_STR);
this->map[elementId] = node; cache.insert(elementId, node);
if (elementId == id) if (elementId == id)
{
return true;
}
}
catch (const VExceptionConversionError &)
{ {
// do nothing return true;
} }
} }
@ -281,7 +281,7 @@ bool VDomDocument::find(const QDomElement &node, quint32 id)
const QDomNode n = node.childNodes().at(i); const QDomNode n = node.childNodes().at(i);
if (n.isElement()) if (n.isElement())
{ {
if (this->find(n.toElement(), id)) if (VDomDocument::find(cache, n.toElement(), id))
{ {
return true; return true;
} }
@ -290,6 +290,15 @@ bool VDomDocument::find(const QDomElement &node, quint32 id)
return false; return false;
} }
//---------------------------------------------------------------------------------------------------------------------
QHash<quint32, QDomElement> VDomDocument::RefreshCache(const QDomElement &root) const
{
QHash<quint32, QDomElement> cache;
VDomDocument::find(cache, root, NULL_ID);
return cache;
}
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
bool VDomDocument::SaveCanonicalXML(QIODevice *file, int indent, QString &error) const bool VDomDocument::SaveCanonicalXML(QIODevice *file, int indent, QString &error) const
{ {
@ -602,6 +611,19 @@ void VDomDocument::CollectId(const QDomElement &node, QVector<quint32> &vector)
} }
} }
//---------------------------------------------------------------------------------------------------------------------
void VDomDocument::RefreshElementIdCache()
{
m_watcher->setFuture(QtConcurrent::run(this, &VDomDocument::RefreshCache, documentElement()));
}
//---------------------------------------------------------------------------------------------------------------------
void VDomDocument::CacheRefreshed()
{
m_elementIdCache = m_watcher->future().result();
m_refreshCacheTimer->start();
}
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------
/** /**
* @brief ValidateXML validate xml file by xsd schema. * @brief ValidateXML validate xml file by xsd schema.
@ -690,6 +712,8 @@ void VDomDocument::setXMLContent(const QString &fileName)
.arg(fileName)); .arg(fileName));
throw e; throw e;
} }
RefreshElementIdCache();
} }
//--------------------------------------------------------------------------------------------------------------------- //---------------------------------------------------------------------------------------------------------------------

View File

@ -50,6 +50,7 @@
class QDomElement; class QDomElement;
class QDomNode; class QDomNode;
template <typename T> class QVector; template <typename T> class QVector;
template <typename T> class QFutureWatcher;
Q_DECLARE_LOGGING_CATEGORY(vXML) Q_DECLARE_LOGGING_CATEGORY(vXML)
@ -78,9 +79,9 @@ QT_WARNING_DISABLE_GCC("-Wnon-virtual-dtor")
* Of these, 2) and 3) are visible in the final pattern (draw mode 'Modeling'), * Of these, 2) and 3) are visible in the final pattern (draw mode 'Modeling'),
* 1) is only displayed when editing (draw mode 'Calculation') the pattern. * 1) is only displayed when editing (draw mode 'Calculation') the pattern.
*/ */
class VDomDocument : public QDomDocument class VDomDocument : public QObject, public QDomDocument
{ {
Q_DECLARE_TR_FUNCTIONS(VDomDocument) Q_OBJECT
public: public:
static const QString AttrId; static const QString AttrId;
static const QString AttrText; static const QString AttrText;
@ -93,7 +94,7 @@ public:
static const QString TagUnit; static const QString TagUnit;
static const QString TagLine; static const QString TagLine;
VDomDocument(); VDomDocument(QObject *parent = nullptr);
virtual ~VDomDocument() Q_DECL_EQ_DEFAULT; virtual ~VDomDocument() Q_DECL_EQ_DEFAULT;
QDomElement elementById(quint32 id, const QString &tagName = QString()); QDomElement elementById(quint32 id, const QString &tagName = QString());
@ -141,12 +142,19 @@ protected:
void TestUniqueId() const; void TestUniqueId() const;
void CollectId(const QDomElement &node, QVector<quint32> &vector)const; void CollectId(const QDomElement &node, QVector<quint32> &vector)const;
private slots:
void RefreshElementIdCache();
void CacheRefreshed();
private: private:
Q_DISABLE_COPY(VDomDocument) Q_DISABLE_COPY(VDomDocument)
/** @brief Map used for finding element by id. */ /** @brief Map used for finding element by id. */
QHash<quint32, QDomElement> map; QHash<quint32, QDomElement> m_elementIdCache;
QTimer *m_refreshCacheTimer;
QFutureWatcher<QHash<quint32, QDomElement>> *m_watcher;
bool find(const QDomElement &node, quint32 id); static bool find(QHash<quint32, QDomElement> &cache, const QDomElement &node, quint32 id);
QHash<quint32, QDomElement> RefreshCache(const QDomElement &root) const;
bool SaveCanonicalXML(QIODevice *file, int indent, QString &error) const; bool SaveCanonicalXML(QIODevice *file, int indent, QString &error) const;
}; };