Qt :Ordered Map

在项目中,有时候对数据结构有这样的需求,既需要具备Map的高效读写,又要兼具插入数据成员的有序性,这时候你就需要使用Ordered Map了。
关于Ordered Map,相关资源比较多,实现思路比较简单,基本上都是通过list有序记录插入的数据的key值,重写遍历或者迭代的方法时,限定按照key的插入顺序遍历。
笔者只是在这里推荐一个现成的开源项目:《Qt Ordered Map》

#ifndef ORDEREDMAP_H
#define ORDEREDMAP_H#include <QtGlobal>
#include <QHash>
#include <QLinkedList>
#include <QList>
#include <QPair>#ifdef Q_COMPILER_INITIALIZER_LISTS
#include <initializer_list>
#endiftemplate <typename Key> inline bool oMHashEqualToKey(const Key &key1, const Key &key2)
{// Key type must provide '==' operatorreturn key1 == key2;
}template <typename Ptr> inline bool oMHashEqualToKey(Ptr *key1, Ptr *key2)
{Q_ASSERT(sizeof(quintptr) == sizeof(Ptr *));return quintptr(key1) == quintptr(key2);
}template <typename Ptr> inline bool oMHashEqualToKey(const Ptr *key1, const Ptr *key2)
{Q_ASSERT(sizeof(quintptr) == sizeof(const Ptr *));return quintptr(key1) == quintptr(key2);
}template <typename Key, typename Value>
class OrderedMap
{class OMHash;typedef typename QLinkedList<Key>::iterator QllIterator;typedef typename QLinkedList<Key>::const_iterator QllConstIterator;typedef QPair<Value, QllIterator> OMHashValue;typedef typename OMHash::iterator OMHashIterator;typedef typename OMHash::const_iterator OMHashConstIterator;public:class iterator;class const_iterator;typedef typename OrderedMap<Key, Value>::iterator Iterator;typedef typename OrderedMap<Key, Value>::const_iterator ConstIterator;explicit OrderedMap();#ifdef Q_COMPILER_INITIALIZER_LISTSOrderedMap(std::initializer_list<std::pair<Key,Value> > list);
#endifOrderedMap(const OrderedMap<Key, Value>& other);#if (QT_VERSION >= 0x050200)OrderedMap(OrderedMap<Key, Value>&& other);
#endifvoid clear();bool contains(const Key &key) const;int count() const;bool empty() const;iterator insert(const Key &key, const Value &value);bool isEmpty() const;QList<Key> keys() const;int remove(const Key &key);int size() const;Value take(const Key &key);Value value(const Key &key) const;Value value(const Key &key, const Value &defaultValue) const;QList<Value> values() const;OrderedMap<Key, Value> & operator=(const OrderedMap<Key, Value>& other);#if (QT_VERSION >= 0x050200)OrderedMap<Key, Value> & operator=(OrderedMap<Key, Value>&& other);
#endifbool operator==(const OrderedMap<Key, Value> &other) const;bool operator!=(const OrderedMap<Key, Value> &other) const;Value& operator[](const Key &key);const Value operator[](const Key &key) const;iterator begin();const_iterator begin() const;iterator end();const_iterator end() const;iterator erase(iterator pos);iterator find(const Key& key);const_iterator find(const Key& key) const;class const_iterator;class iterator{QllIterator qllIter;OMHash *data;friend class const_iterator;friend class OrderedMap;public:iterator() : data(NULL) {}iterator(const QllIterator &qllIter, OMHash *data) :qllIter(qllIter), data(data) {}const Key & key() const{return *qllIter;}Value & value() const{OMHashIterator hit = data->find(*qllIter);OMHashValue &pair = hit.value();return pair.first;}Value & operator*() const{return value();}iterator operator+(int i) const{QllIterator q = qllIter;q += i;return iterator(q, data);}iterator operator-(int i) const{return operator +(- i);}iterator& operator+=(int i){qllIter += i;return *this;}iterator& operator-=(int i){qllIter -= i;return *this;}iterator& operator++(){++qllIter;return *this;}iterator operator++(int){iterator it = *this;qllIter++;return it;}iterator operator--(){--qllIter;return *this;}iterator operator--(int){iterator it = *this;qllIter--;return it;}bool operator ==(const iterator &other) const{return (qllIter == other.qllIter);}bool operator !=(const iterator &other) const{return (qllIter != other.qllIter);}};class const_iterator{QllConstIterator qllConstIter;const OMHash *data;public:const_iterator() : data(NULL) {}const_iterator(const iterator &i) :qllConstIter(i.qllIter), data(i.data) {}const_iterator(const QllConstIterator &qllConstIter, const OMHash* data) :qllConstIter(qllConstIter), data(data) {}const Key & key() const{return *qllConstIter;}const Value & value() const{OMHashConstIterator hit = data->find(*qllConstIter);const OMHashValue &pair = hit.value();return pair.first;}const Value & operator*() const{return value();}const_iterator operator+(int i) const{QllConstIterator q = qllConstIter;q += i;return const_iterator(q, data);}const_iterator operator-(int i) const{return operator +(- i);}const_iterator& operator+=(int i){qllConstIter += i;return *this;}const_iterator& operator-=(int i){qllConstIter -= i;return *this;}const_iterator& operator++(){++qllConstIter;return *this;}const_iterator operator++(int){const_iterator it = *this;qllConstIter++;return it;}const_iterator operator--(){--qllConstIter;return *this;}const_iterator operator--(int){const_iterator it = *this;qllConstIter--;return it;}bool operator ==(const const_iterator &other) const{return (qllConstIter == other.qllConstIter);}bool operator !=(const const_iterator &other) const{return (qllConstIter != other.qllConstIter);}};private:class OMHash : public QHash<Key, OMHashValue >{public:bool operator == (const OMHash &other) const{if (size() != other.size()) {return false;}if (QHash<Key, OMHashValue >::operator ==(other)) {return true;}typename QHash<Key, OMHashValue >::const_iterator it1 = this->constBegin();typename QHash<Key, OMHashValue >::const_iterator it2 = other.constBegin();while(it1 != this->end()) {OMHashValue v1 = it1.value();OMHashValue v2 = it2.value();if ((v1.first != v2.first) || !oMHashEqualToKey<Key>(it1.key(), it2.key())) {return false;}++it1;++it2;}return true;}};private:void copy(const OrderedMap<Key, Value> &other);OMHash data;QLinkedList<Key> insertOrder;
};template <typename Key, typename Value>
OrderedMap<Key, Value>::OrderedMap() {}#ifdef Q_COMPILER_INITIALIZER_LISTS
template<typename Key, typename Value>
OrderedMap<Key, Value>::OrderedMap(std::initializer_list<std::pair<Key, Value> > list)
{typedef typename std::initializer_list<std::pair<Key,Value> >::const_iterator const_initlist_iter;for (const_initlist_iter it = list.begin(); it != list.end(); ++it)insert(it->first, it->second);
}
#endiftemplate <typename Key, typename Value>
OrderedMap<Key, Value>::OrderedMap(const OrderedMap<Key, Value>& other)
{copy(other);
}#if (QT_VERSION >= 0x050200)
template <typename Key, typename Value>
OrderedMap<Key, Value>::OrderedMap(OrderedMap<Key, Value>&& other)
{data = std::move(other.data);insertOrder = std::move(other.insertOrder);
}
#endiftemplate <typename Key, typename Value>
void OrderedMap<Key, Value>::clear()
{data.clear();insertOrder.clear();
}template <typename Key, typename Value>
bool OrderedMap<Key, Value>::contains(const Key &key) const
{return data.contains(key);
}template <typename Key, typename Value>
int OrderedMap<Key, Value>::count() const
{return data.count();
}template <typename Key, typename Value>
bool OrderedMap<Key, Value>::empty() const
{return data.empty();
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::iterator OrderedMap<Key, Value>::insert(const Key &key, const Value &value)
{OMHashIterator it = data.find(key);if (it == data.end()) {// New keyQllIterator ioIter = insertOrder.insert(insertOrder.end(), key);OMHashValue pair(value, ioIter);data.insert(key, pair);return iterator(ioIter, &data);}OMHashValue pair = it.value();// remove old referenceinsertOrder.erase(pair.second);// Add new referenceQllIterator ioIter = insertOrder.insert(insertOrder.end(), key);pair.first = value;pair.second = ioIter;data.insert(key, pair);return iterator(ioIter, &data);
}template <typename Key, typename Value>
bool OrderedMap<Key, Value>::isEmpty() const
{return data.isEmpty();
}template<typename Key, typename Value>
QList<Key> OrderedMap<Key, Value>::keys() const
{return QList<Key>::fromStdList(insertOrder.toStdList());
}template<typename Key, typename Value>
int OrderedMap<Key, Value>::remove(const Key &key)
{OMHashIterator it = data.find(key);if (it == data.end()) {return 0;}OMHashValue pair = it.value();insertOrder.erase(pair.second);data.erase(it);return 1;
}template<typename Key, typename Value>
int OrderedMap<Key, Value>::size() const
{return data.size();
}template<typename Key, typename Value>
void OrderedMap<Key, Value>::copy(const OrderedMap<Key, Value> &other)
{/* Since I'm storing iterators of QLinkedList, I simply cannot make* a trivial copy of the linked list. This is a limitation due to implicit* sharing used in Qt containers, due to which iterator active on one* QLL can change the data of another QLL even after creating a copy.** Because of this, the old iterators have to be invalidated and new ones* have to be generated.*/insertOrder.clear();// Copy hashdata = other.data;QllConstIterator cit = other.insertOrder.begin();for (; cit != other.insertOrder.end(); ++cit) {Key key = *cit;QllIterator ioIter = insertOrder.insert(insertOrder.end(), key);OMHashIterator it = data.find(key);(*it).second = ioIter;}
}template<typename Key, typename Value>
Value OrderedMap<Key, Value>::take(const Key &key)
{OMHashIterator it = data.find(key);if (it == data.end()) {return Value();}OMHashValue pair = it.value();insertOrder.erase(pair.second);data.erase(it);return pair.first;
}template <typename Key, typename Value>
Value OrderedMap<Key, Value>::value(const Key &key) const
{return data.value(key).first;
}template <typename Key, typename Value>
Value OrderedMap<Key, Value>::value(const Key &key, const Value &defaultValue) const
{OMHashConstIterator it = data.constFind(key);if (it == data.end()) {return defaultValue;}OMHashValue pair = it.value();return pair.first;
}template <typename Key, typename Value>
QList<Value> OrderedMap<Key, Value>::values() const
{QList<Value> values;foreach (const Key &key, insertOrder.toStdList()) {OMHashValue v = data.value(key);values.append(v.first);}return values;
}template <typename Key, typename Value>
OrderedMap<Key, Value> & OrderedMap<Key, Value>::operator=(const OrderedMap<Key, Value>& other)
{if (this != &other) {copy(other);}return *this;
}#if (QT_VERSION >= 0x050200)
template <typename Key, typename Value>
OrderedMap<Key, Value> & OrderedMap<Key, Value>::operator=(OrderedMap<Key, Value>&& other)
{if (this != &other) {data = other.data;insertOrder = other.insertOrder;}return *this;
}
#endiftemplate <typename Key, typename Value>
bool OrderedMap<Key, Value>::operator==(const OrderedMap<Key, Value> &other) const
{// 2 Ordered maps are equal if they have the same contents in the same orderreturn ((data == other.data) && (insertOrder == other.insertOrder));
}template <typename Key, typename Value>
bool OrderedMap<Key, Value>::operator!=(const OrderedMap<Key, Value> &other) const
{return ((data != other.data) || (insertOrder != other.insertOrder));
}template <typename Key, typename Value>
Value& OrderedMap<Key, Value>::operator[](const Key &key)
{OMHashIterator it = data.find(key);if (it == data.end()) {insert(key, Value());it = data.find(key);}OMHashValue &pair = it.value();return pair.first;
}template <typename Key, typename Value>
const Value OrderedMap<Key, Value>::operator[](const Key &key) const
{return value(key);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::iterator OrderedMap<Key, Value>::begin()
{return iterator(insertOrder.begin(), &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::const_iterator OrderedMap<Key, Value>::begin() const
{return const_iterator(insertOrder.begin(), &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::iterator OrderedMap<Key, Value>::end()
{return iterator(insertOrder.end(), &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::const_iterator OrderedMap<Key, Value>::end() const
{return const_iterator(insertOrder.end(), &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::iterator OrderedMap<Key, Value>::erase(iterator pos)
{OMHashIterator hit = data.find(*(pos.qllIter));if (hit == data.end()) {return pos;}data.erase(hit);QllIterator ioIter = insertOrder.erase(pos.qllIter);return iterator(ioIter, &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::iterator OrderedMap<Key, Value>::find(const Key& key)
{OMHashIterator hit = data.find(key);if (hit == data.end()) {return end();}return iterator(hit.value().second, &data);
}template <typename Key, typename Value>
typename OrderedMap<Key, Value>::const_iterator OrderedMap<Key, Value>::find(const Key& key) const
{OMHashConstIterator hit = data.find(key);if (hit == data.end()) {return end();}return const_iterator(hit.value().second, &data);
}
#endif // ORDEREDMAP_H

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/3452.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用零拷贝技术加速视频图像操作

不使用cv::cuda::HostMem的情况 当直接创建cv::cuda::GpuMat并从CPU内存拷贝数据到GPU时&#xff0c;如果没有共享内存机制&#xff0c;数据传输是通过常规的内存拷贝完成的&#xff0c;例如&#xff1a; uchar* nv12Data; // CPU内存中的NV12数据 cv::Mat hostImage(height …

C# 获取一个字符串中数字部分?

要在 C# 中将一个字符串中只保留数字字符&#xff0c;以下是两种实现方法&#xff1a; 方法一&#xff1a;使用正则表达式 正则表达式是一种强大的文本模式匹配工具&#xff0c;可以轻松地提取字符串中符合特定规则的部分。 在这种情况下&#xff0c;我们可以使用一个匹配数…

数据挖掘实验一

一、实验环境及背景 使用软件&#xff1a; Anaconda3 Jupyter Notebook 实验内容&#xff1a; 1.使用Tushare或者其他手段获取任意两支股票近三个月的交易数据。做出收盘价的变动图像。2.使用Pandas_datareader获取世界银行数据库中美国&#xff08;USA&#xff09;、瑞典&…

【中级软件设计师】上午题12-软件工程(2):单元测试、黑盒测试、白盒测试、软件运行与维护

【中级软件设计师】上午题12-软件工程&#xff08;2&#xff09; 1 系统测试1.1 单元测试1.2 集成测试1.2.1 自顶向下1.2.2 自顶向上1.2.3 回归测试 2 测试方法2.1 黑盒测试2.1.1 McCabe度量法 2.2 白盒测试2.2.1 语句覆盖-“每个流程”执行一次2.2.2 判定覆盖2.2.3 条件覆盖-A…

资源、死锁、如何监测死锁

资源&#xff1a;变量、对象、网络资源、文件、内存、cpu等等 &#xff08;应用层或者会话层 资源&#xff09; 统一资源限定符&#xff1a;[scheme:][//authority][path][?query][#fragment] authority 可以表示&#xff1a; [user-info]host[:port] 比如&#xff1a; fi…

vue+springboot项目的登录验证码(JAVA自带)

后台springboot CaptureController package com.example.controller;import com.example.common.Result; import com.example.service.AuthCodeService; import com.example.utils.CodeUtils; import lombok.SneakyThrows; import org.apache.ibatis.annotations.Param; impo…

Java openrasp记录-01

例子1 https://github.com/anbai-inc/javaweb-expression 一个hook ognl、spel、MVEL表达式注入的例子 用的是asm5进行字节码修改 采用premain进行插桩&#xff0c;重写transform方法 expClassList是要hook的类&#xff0c;这里定义在MethodHookDesc 这里判断hook点通过类名…

Java并发编程:ThreadPoolExecutor详细源码解析与应用

Thread直接创建线程的弊端 在开始解析ThreadPoolExecutor类之前&#xff0c;让我们先来了解直接创建线程所带来的弊端。在Java中&#xff0c;线程是资源密集型对象&#xff0c;每当需要并发执行任务时&#xff0c;直接创建新线程会带来以下问题&#xff1a; 资源消耗 每个线…

贪吃蛇身子改进加贪吃蛇向右移动

1. 蛇移动的思想&#xff1a; 其实就是删除头节点 &#xff0c;增加尾节点&#xff1b;一句代码搞定 struct Snake *p; p head; head head -> next; free(p) 防止造成多的空间节点 2.增加尾节点代码思想&#xff1a; 2.1 .开辟new 节点的空间 struct Snake *new (stru…

解决Android studio更换sdk地址后flutter项目显示no device selected

问题描述 因为之前sdk的路径在c盘上&#xff0c;经常在更新或下在sdk后c盘饱满&#xff0c;于是就更换了sdk的路径&#xff0c;更换sdk路径后就导致flutter项目在选择设备的时候出现no device selected 找不到设备&#xff0c;但是在device Manager可以看到物理设备或者是虚拟…

深入了解Eureka:微服务架构中的服务发现与注册中心

引言 微服务架构的兴起使得应用程序变得更加模块化和可扩展。在这种架构下&#xff0c;服务发现与注册中心扮演着至关重要的角色。本文将深入探讨Eureka作为服务发现与注册中心的作用、优缺点、重要性以及其服务架构。 一、Eureka的作用 Eureka是Netflix开源的一款用于构建分…

【Quartz】quartz集群模式下数据库表设计原则(详细版)

目录 引言 1. 数据共享性 2. 锁管理 3. 持久性和一致性 4. 状态追踪 5. 元数据管理 6. 优化和索引 主要的Quartz表包括 引言 Quartz 是一个流行的开源作业调度库&#xff0c;广泛用于在Java应用程序中安排作业。在Quartz的集群模式中&#xff0c;多个Quartz实例共享一个…

Java设计模式:使用责任链模式和状态模式优化‘审批流程‘

Java设计模式&#xff1a;使用责任链模式和状态模式优化审批流程 摘要引言 需求流程图正文内容&#x1f4d0; 基本概念介绍 功能实现示例1:设计模式&#xff1a;责任链模式方法&#xff1a;好处&#xff1a; 示例2:设计模式&#xff1a;责任链模式方法和操作流程&#xff1a;好…

【canvas】前端创造的图片粒子动画效果:HTML5 Canvas 技术详解

前端创造的图片粒子动画效果&#xff1a;HTML5 Canvas 技术详解 我们将深入探讨如何通过 HTML5 的 Canvas 功能&#xff0c;将上传的图片转换成引人入胜的粒子动画效果。这种效果将图片分解成小粒子&#xff0c;并在用户与它们交互时产生动态变化。我们将分步骤详细解析代码&a…

EasyRecovery数据恢复软件2025永久免费电脑版下载

EasyRecovery数据恢复软件是一款业界知名的数据恢复工具&#xff0c;它凭借强大的恢复能力和广泛的数据兼容性&#xff0c;帮助用户从各种存储设备中恢复丢失或删除的数据。以下是关于EasyRecovery数据恢复软件的详细介绍。 EasyRecovery绿色破解下载网盘链接: https://pan.ba…

自动驾驶行业源代码防泄漏解决方案

行业背景&#xff1a; 随着新一代信息通信及人工智能技术的快速发展&#xff0c;汽车作为这些新技术应用的重要载体&#xff0c;正在加速向智能化和网联化转型&#xff0c;以自动驾驶研发为主业的企业也越来越多&#xff0c;如何保障自己研发的算法、模型、系统不被研发人员离…

Linux入门攻坚——20、systemd、(sysvinit、upstart重温)

再一次讲到Linux系统启动流程&#xff1a; POST --> Boot Sequence --> Bootloader(grub) --> kernel initramfs(initrd) --> rootfs --> /sbin/init 对于init&#xff0c;即系统内核加载完毕后&#xff08;加载kernel和切换根文件系统&#xff09;运行…

Python项目开发实战:怎么实现端口扫描器

注意:本文的下载教程,与以下文章的思路有相同点,也有不同点,最终目标只是让读者从多维度去熟练掌握本知识点。 下载教程:Python项目开发实战_端口扫描器的实现_编程案例解析实例详解课程教程.pdf 1、步骤 在Python项目开发中,设计并实现一个端口扫描器是一项基础且实用的…

【WebRTC】【Unity】局域网UDP通信为何不通

【背景】 还是在研究Unity中实现VR桌面&#xff0c;希望能够通过UDP广播先找到所有活跃的Client。但是发现UDP广播并未能够成功传递给同一局域网正在运行的客户端。 【分析】 UDP信息在局域网不通可能有如下几个原因&#xff1a; 未连在同一个网段防火墙问题是否存在其它网…

STM32H750外设ADC之开始和结束数据转换功能

目录 概述 1 开始转换 1.1 使能ADSTART 1.2 使能JADSTART 1.3 ADSTART 通过硬件清零 2 转换时序 3 停止正在进行的转换&#xff08; ADSTP、 JADSTP&#xff09; 3.1 停止转换功能实现 3.2 停止转换流程图 概述 本文主要讲述了STM32H750外设ADC之开始和结束数据转换…