123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- /****************************************************************************
- **
- ** 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 >= 0x050600)
- 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 >= 0x050600)
- 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 >= 0x050600)
- 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 >= 0x050600)
- 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_
|