You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
282 lines
15 KiB
282 lines
15 KiB
3 days ago
|
/****************************************************************************
|
||
|
**
|
||
|
** https://www.qxorm.com/
|
||
|
** Copyright (C) 2013 Lionel Marty (contact@qxorm.com)
|
||
|
**
|
||
|
** This file is part of the QxOrm library
|
||
|
**
|
||
|
** This software is provided 'as-is', without any express or implied
|
||
|
** warranty. In no event will the authors be held liable for any
|
||
|
** damages arising from the use of this software
|
||
|
**
|
||
|
** Commercial Usage
|
||
|
** Licensees holding valid commercial QxOrm licenses may use this file in
|
||
|
** accordance with the commercial license agreement provided with the
|
||
|
** Software or, alternatively, in accordance with the terms contained in
|
||
|
** a written agreement between you and Lionel Marty
|
||
|
**
|
||
|
** GNU General Public License Usage
|
||
|
** Alternatively, this file may be used under the terms of the GNU
|
||
|
** General Public License version 3.0 as published by the Free Software
|
||
|
** Foundation and appearing in the file 'license.gpl3.txt' included in the
|
||
|
** packaging of this file. Please review the following information to
|
||
|
** ensure the GNU General Public License version 3.0 requirements will be
|
||
|
** met : http://www.gnu.org/copyleft/gpl.html
|
||
|
**
|
||
|
** If you are unsure which license is appropriate for your use, or
|
||
|
** if you have questions regarding the use of this file, please contact :
|
||
|
** contact@qxorm.com
|
||
|
**
|
||
|
****************************************************************************/
|
||
|
|
||
|
#ifndef _QX_COLLECTION_H_
|
||
|
#define _QX_COLLECTION_H_
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma once
|
||
|
#endif
|
||
|
|
||
|
/*!
|
||
|
* \file QxCollection.h
|
||
|
* \author Lionel Marty
|
||
|
* \ingroup QxCollection
|
||
|
* \brief QxOrm thread-safe container (keep insertion order + quick access by index + quick access by key)
|
||
|
*/
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma warning(push)
|
||
|
#pragma warning(disable:4996)
|
||
|
#pragma warning(disable:4503)
|
||
|
#endif // _MSC_VER
|
||
|
|
||
|
#include <QtCore/qmutex.h>
|
||
|
|
||
|
#include <QxCollection/IxCollection.h>
|
||
|
#include <QxCollection/QxForeach.h>
|
||
|
|
||
|
#include <QxCommon/QxHashValue.h>
|
||
|
|
||
|
#include <QxTraits/get_class_name.h>
|
||
|
#include <QxTraits/is_smart_ptr.h>
|
||
|
|
||
|
namespace qx {
|
||
|
|
||
|
/*!
|
||
|
* \ingroup QxCollection
|
||
|
* \brief qx::QxCollection<Key, Value> : QxOrm thread-safe container (keep insertion order + quick access by index + quick access by key)
|
||
|
*
|
||
|
* Based on boost::multi_index_container, this collection has advantages of std::vector<T> (keep insertion order + quick access by index)
|
||
|
* and boost::unordered_map<Key, Value> or QHash<Key, Value> (quick access by key : hash-map).
|
||
|
*
|
||
|
* <i>Note :</i> qx::QxCollection<Key, Value> is compatible with the foreach macro provided by Qt library and the BOOST_FOREACH macro provided by boost library.
|
||
|
* However, each element returned by these 2 macros corresponds to an object of type std::pair<Key, Value>.
|
||
|
* To obtain a more natural and more readable result, it is advised to use the _foreach macro : this macro uses BOOST_FOREACH for all the containers except for qx::QxCollection<Key, Value>.
|
||
|
* In this case, the returned element corresponds to the Value type (cf. following sample).
|
||
|
* The macro _foreach is compatible with all containers (stl, Qt, boost...) since it uses the macro BOOST_FOREACH.
|
||
|
*
|
||
|
* <i>Additional note :</i> qx::QxCollection<Key, Value> is particularly suited to receive data resulting from a database.
|
||
|
* Indeed, these data can be sorted (by using ORDER BY in a SQL request for example), it is thus important to preserve the insertion order of the elements in the list.
|
||
|
* Furthermore, each data resulting from a database has a unique id. It is thus important to be able to access quickly to an element based on this single identifier (hash-map).
|
||
|
*
|
||
|
* Quick sample using qx::QxCollection<Key, Value> container :
|
||
|
* \code
|
||
|
// definition of drug class with 3 properties : 'code', 'name' and 'description'
|
||
|
class drug { public: QString code; QString name; QString desc; };
|
||
|
|
||
|
// typedef a smart-pointer of drug class
|
||
|
typedef std::shared_ptr<drug> drug_ptr;
|
||
|
|
||
|
// collection of drugs indexed by 'code' property (QString type)
|
||
|
qx::QxCollection<QString, drug_ptr> lstDrugs;
|
||
|
|
||
|
// create 3 new drugs
|
||
|
drug_ptr d1; d1.reset(new drug()); d1->code = "code1"; d1->name = "name1"; d1->desc = "desc1";
|
||
|
drug_ptr d2; d2.reset(new drug()); d2->code = "code2"; d2->name = "name2"; d2->desc = "desc2";
|
||
|
drug_ptr d3; d3.reset(new drug()); d3->code = "code3"; d3->name = "name3"; d3->desc = "desc3";
|
||
|
|
||
|
// insert 3 drugs into the collection
|
||
|
lstDrugs.insert(d1->code, d1);
|
||
|
lstDrugs.insert(d2->code, d2);
|
||
|
lstDrugs.insert(d3->code, d3);
|
||
|
|
||
|
// iterate over drugs container using QxOrm '_foreach' keyword
|
||
|
_foreach(drug_ptr p, lstDrugs)
|
||
|
{ qDebug() << qPrintable(p->name) << " " << qPrintable(p->desc); }
|
||
|
|
||
|
// iterate over drugs container using classic C++ 'for' keyword
|
||
|
for (long l = 0; l < lstDrugs.count(); ++l)
|
||
|
{
|
||
|
drug_ptr p = lstDrugs.getByIndex(l);
|
||
|
QString code = lstDrugs.getKeyByIndex(l);
|
||
|
qDebug() << qPrintable(p->name) << " " << qPrintable(p->desc);
|
||
|
}
|
||
|
|
||
|
// iterate over drugs container using 'qx::QxCollectionIterator' Java-style iterator
|
||
|
qx::QxCollectionIterator<QString, drug_ptr> itr(lstDrugs);
|
||
|
while (itr.next())
|
||
|
{
|
||
|
QString code = itr.key();
|
||
|
qDebug() << qPrintable(itr.value()->name) << " " << qPrintable(itr.value()->desc);
|
||
|
}
|
||
|
|
||
|
// sort drugs container ascending by key and sort descending by value
|
||
|
lstDrugs.sortByKey(true);
|
||
|
lstDrugs.sortByValue(false);
|
||
|
|
||
|
// access to a drug into the collection by its 'code' property
|
||
|
drug_ptr p = lstDrugs.getByKey("code2");
|
||
|
|
||
|
// access to a drug into the collection by index (position in the list)
|
||
|
drug_ptr p = lstDrugs.getByIndex(2);
|
||
|
|
||
|
// test if a drug exists into the collection and if the collection is empty
|
||
|
bool bExist = lstDrugs.exist("code3");
|
||
|
bool bEmpty = lstDrugs.empty();
|
||
|
|
||
|
// remove the second drug from the collection
|
||
|
lstDrugs.removeByIndex(2);
|
||
|
|
||
|
// remove a drug from the collection using its 'code' property
|
||
|
lstDrugs.removeByKey("code3");
|
||
|
|
||
|
// clear the collection : remove all items from the list
|
||
|
lstDrugs.clear();
|
||
|
* \endcode
|
||
|
*/
|
||
|
template <typename Key, typename Value>
|
||
|
class QxCollection : public IxCollection
|
||
|
{
|
||
|
|
||
|
public:
|
||
|
|
||
|
typedef QPair<Key, Value> type_pair_key_value;
|
||
|
|
||
|
protected:
|
||
|
|
||
|
typedef QList<type_pair_key_value> type_list_pair_key_value;
|
||
|
typedef QHash<Key, long> type_hash_position;
|
||
|
|
||
|
public:
|
||
|
|
||
|
typedef typename type_list_pair_key_value::iterator iterator;
|
||
|
typedef typename type_list_pair_key_value::const_iterator const_iterator;
|
||
|
|
||
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
|
||
|
typedef typename type_list_pair_key_value::reverse_iterator reverse_iterator;
|
||
|
typedef typename type_list_pair_key_value::const_reverse_iterator const_reverse_iterator;
|
||
|
#endif // (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
|
||
|
|
||
|
typedef const Key & const_reference_key;
|
||
|
typedef const Value & const_reference_value;
|
||
|
|
||
|
protected:
|
||
|
|
||
|
mutable QMutex m_mutex; //!< Mutex => qx::QxCollection is thread-safe
|
||
|
type_list_pair_key_value m_list; //!< Container to keep insertion order
|
||
|
type_hash_position m_hash; //!< Container for fast search by key
|
||
|
bool m_batch; //!< Batch mode to sync internal containers
|
||
|
|
||
|
public:
|
||
|
|
||
|
QxCollection(); //!< Construct an empty list
|
||
|
QxCollection(const QxCollection<Key, Value> & other); //!< Construct a copy of 'other'
|
||
|
virtual ~QxCollection(); //!< Destroy the list
|
||
|
|
||
|
QxCollection<Key, Value> & operator= (const QxCollection<Key, Value> & other); //!< Assign 'other' to this list and return a reference to this list
|
||
|
bool operator== (const QxCollection<Key, Value> & other) const; //!< Return 'true' if 'other' is equal to this list, otherwise return 'false' (same values in the same order)
|
||
|
bool operator!= (const QxCollection<Key, Value> & other) const; //!< Return 'true' if 'other' is not equal to this list, otherwise return 'false'
|
||
|
|
||
|
iterator begin(); //!< Return an STL-style iterator pointing to the first item in the list
|
||
|
iterator end(); //!< Return an STL-style iterator pointing to the imaginary item after the last item in the list
|
||
|
const_iterator begin() const; //!< Return a const STL-style iterator pointing to the first item in the list
|
||
|
const_iterator end() const; //!< Return a const STL-style iterator pointing to the imaginary item after the last item in the list
|
||
|
|
||
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
|
||
|
reverse_iterator rbegin(); //!< Return a reverse STL-style iterator pointing to the first item in the list
|
||
|
reverse_iterator rend(); //!< Return a reverse STL-style iterator pointing to the imaginary item after the last item in the list
|
||
|
const_reverse_iterator rbegin() const; //!< Return a const reverse STL-style iterator pointing to the first item in the list
|
||
|
const_reverse_iterator rend() const; //!< Return a const reverse STL-style iterator pointing to the imaginary item after the last item in the list
|
||
|
#endif // (QT_VERSION >= QT_VERSION_CHECK(5, 6, 0))
|
||
|
|
||
|
void reserve(long size); //!< Request that the capacity of the allocated storage space for the items of the container be at least enough to hold 'size' elements
|
||
|
void reverse(); //!< Reverse all items in the list
|
||
|
void clear(); //!< Remove all items from the list
|
||
|
long count() const; //!< Return the number of items in the list (same as 'size()')
|
||
|
long size() const; //!< Return the number of items in the list (same as 'count()')
|
||
|
bool contains(const Key & key) const; //!< Return 'true' if the list contains an occurrence of 'key', otherwise return 'false' (same as 'exist()')
|
||
|
bool exist(const Key & key) const; //!< Return 'true' if the list contains an occurrence of 'key', otherwise return 'false' (same as 'contains()')
|
||
|
bool empty() const; //!< Return 'true' if the list contains no items; otherwise return 'false'
|
||
|
|
||
|
bool push_back(const Key & key, const Value & value); //!< Add element 'value' at the end of the list indexed by 'key'
|
||
|
bool push_front(const Key & key, const Value & value); //!< Insert 'value' at the beginning of the list indexed by 'key'
|
||
|
bool insert(const Key & key, const Value & value); //!< Add element 'value' at the end of the list indexed by 'key'
|
||
|
bool insert(long index, const Key & key, const Value & value); //!< Insert element 'value' at position 'index' in the list indexed by 'key'
|
||
|
bool insert(const QxCollection<Key, Value> & other); //!< Add all items of 'other' at the end of the list
|
||
|
bool insert(long index, const QxCollection<Key, Value> & other); //!< Insert all items of 'other' at the end of the list
|
||
|
bool replace(long index, const Key & key, const Value & value); //!< Replace the item at index position 'index' with element 'value' indexed by 'key'
|
||
|
bool swap(long index1, long index2); //!< Exchange the item at index position 'index1' with the item at index position 'index2'
|
||
|
bool move(long indexFrom, long indexTo); //!< Move the item at index position 'indexFrom' to index position 'indexTo'
|
||
|
|
||
|
bool removeByKey(const Key & key); //!< Remove the item indexed by 'key' in the list
|
||
|
bool removeByIndex(long index); //!< Remove the item at index position 'index'
|
||
|
bool removeByIndex(long first, long last); //!< Remove all items from index position 'first' to index position 'last'
|
||
|
bool removeFirst(); //!< Remove the first item in the list
|
||
|
bool removeLast(); //!< Remove the last item in the list
|
||
|
|
||
|
const_reference_value getByKey(const Key & key) const; //!< Return the item associated with the 'key'
|
||
|
const_reference_value getByIndex(long index) const; //!< Return the item at index position 'index'
|
||
|
const_reference_value getFirst() const; //!< Return the first element in the list
|
||
|
const_reference_value getLast() const; //!< Return the last element in the list
|
||
|
const_reference_key getKeyByIndex(long index) const; //!< Return the key associated with the element at index position 'index'
|
||
|
|
||
|
void sortByKey(bool bAscending = true); //!< Sort all items in the list using associated keys to compare
|
||
|
void sortByValue(bool bAscending = true); //!< Sort all items in the list
|
||
|
|
||
|
template <typename Compare>
|
||
|
void sort(Compare comp) { { QMutexLocker locker(& m_mutex); std::sort(m_list.begin(), m_list.end(), comp); } updateHashPosition(); }
|
||
|
|
||
|
protected:
|
||
|
|
||
|
void cloneCollection(QxCollection<Key, Value> * pClone, const QxCollection<Key, Value> & pRef);
|
||
|
bool isSameCollection(const QxCollection<Key, Value> * p1, const QxCollection<Key, Value> & p2) const;
|
||
|
void updateHashPosition(long from = 0, long to = -1, bool check = false);
|
||
|
|
||
|
template <bool bIsPointer /* = false */, int dummy>
|
||
|
struct compareKeyValue
|
||
|
{
|
||
|
static bool compareByKeyAscending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return (v1.first < v2.first); }
|
||
|
static bool compareByKeyDescending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return (v1.first > v2.first); }
|
||
|
static bool compareByValueAscending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return (v1.second < v2.second); }
|
||
|
static bool compareByValueDescending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return (v1.second > v2.second); }
|
||
|
};
|
||
|
|
||
|
template <int dummy>
|
||
|
struct compareKeyValue<true, dummy>
|
||
|
{
|
||
|
static bool compareByKeyAscending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return ((v1.first && v2.first) ? ((* v1.first) < (* v2.first)) : false); }
|
||
|
static bool compareByKeyDescending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return ((v1.first && v2.first) ? ((* v1.first) > (* v2.first)) : true); }
|
||
|
static bool compareByValueAscending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return ((v1.second && v2.second) ? ((* v1.second) < (* v2.second)) : false); }
|
||
|
static bool compareByValueDescending(const type_pair_key_value & v1, const type_pair_key_value & v2) { return ((v1.second && v2.second) ? ((* v1.second) > (* v2.second)) : true); }
|
||
|
};
|
||
|
|
||
|
public:
|
||
|
|
||
|
virtual long _count() const { return this->count(); }
|
||
|
virtual void _clear() { this->clear(); }
|
||
|
virtual bool _remove(long index) { return this->removeByIndex(index); }
|
||
|
virtual qx::any _at(long index) const { Value val = this->getByIndex(index); return qx::any(val); }
|
||
|
|
||
|
};
|
||
|
|
||
|
} // namespace qx
|
||
|
|
||
|
#include "../../inl/QxCollection/QxCollection.inl"
|
||
|
|
||
|
QX_REGISTER_CLASS_NAME_TEMPLATE_2(qx::QxCollection)
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma warning(pop)
|
||
|
#endif // _MSC_VER
|
||
|
|
||
|
#endif // _QX_COLLECTION_H_
|