拇指接龙游戏中的Undo道具与STL容器deque简介

在线体验各类最新模型,更有模型 免费Token 额度领取!
立即体验
简介:

引子

在我的视频课程《拇指姑娘空当接龙》游戏(http://edu.51cto.com/course/course_id-1830.html)中,Undo道具对于游戏新手来说应当是最常用道具之一。其作用类似于下棋中的“悔棋”。因为接龙游戏要求玩家在移牌前要先进行心中分析,但是由于此游戏的复杂性(特别是到了后期较复杂的回合布局中),由于玩家预先分析不周密,可能导致需要“悔牌”操作;否则,整个牌局将陷于死局中(当然,如果红心数足够,玩家也可以借助其他道具解救)。


  需求

上述Undo道具(“悔牌”操作)涉及到如下一些情况:


  • 选择什么样的数据结构实现最恰当?

  • 可以悔一张牌

  • 可以悔系列牌(即玩家把一个系列从下部区域的某一列移动到另一列时)

  • 可以连续多次悔牌

  • 悔牌后对应顺序与位置不可改变

  • 最多支持多少次连续悔牌最合适


方案

  基于上述需要,我选择了使用了STL库的双端队列--容器deque (double-ended queue)。我创建了两个deque:一个是主要容器,另一个起辅助标记作用。

  

1
2
     std::deque<UndoNode> UndoDeque; //main
     std::deque< int > AuxUndoDeque; //auxiliary


  我们的悔牌机制具体描述如下:

  目前的设计目标是队列中最多容纳8项,但是这些入队列的项可能有的是单项,有的是小系列(需要一起操作)。


    考虑到有的系列特别长(例如从13点到5点的仅仅一个系列就包含9张牌),作进一步简化处理,方案是:

  如果有一个系列入队列,则先清空队列中所有内容,再入队列此系列。


    为此,我们设计另一个辅助队列AuxUndoDeque,用于记忆撤消队列中可能成组的项。

  算法描述如下:


    i.PUSH当前要入队列的项(单或者系列); 同时,PUSH到AuxUndoDeque中相应的标志int
    ii.如果当前仅有一张扑克入队列,而且如果队列OK(<=8) ,那么情况简单;如果>8(此前队列内总数已为8),则判断如下:
     a. if arr[0]是一个入队列单项,弹出这个单项即可
     b. arr[0..k]是一个小系列,则弹出这个系列。
    iii.如果当前入队列的是一个系列,则复杂些,因为可能需要从队列尾弹出不止一组(多个单项和多个小系列)。
     如果上述辅助队列中对应参数为-1,说明要从尾部弹出一项;如果是N(>1的整数),说明要从尾部弹出N项(小系列)。


  提示:

  1,无论主队列是出队操作(从头部)还是入队操作(从尾部),相应地,其辅助队列也要进行相应的出队和入队操作。

  2,辅助队列中存储一个整数,当-1时表示主队列中加入的是一个单项,如果是N>1的一个整数,则此整数表示主队列中加入的是一个长度为N的系列。


补充-STL容器主要操作知识


deque双向队列是一种双向开口的连续线性空间,可以高效的在头尾两端插入和删除元素,deque在接口上和vector非常相似,下面列出deque的常用成员函数:

头文件:#include<deque>

  

函数构造:

          deque<Elem> c     建一个空的deque。

          deque<Elem> c1(c2)     复制一个deque。

          deque<Elem> c(n)     创建一个deque,含有n个数据,数据均已缺省构造产生。

          deque<Elem> c(n, elem)    创建一个含有n个elem拷贝的deque

          deque<Elem> c(beg,end)     创建一个以[beg;end)区间的deque

          c.~deque<Elem>()     销毁所有数据,释放内存

成员函数:

          c.assign(beg,end)    将[beg; end)区间中的数据赋值给c。

          c.assign(n,elem)    将n个elem的拷贝赋值给c。

          c. at(idx)     传回索引idx所指的数据,如果idx越界,抛出out_of_range。

          c.back()     传回最后一个数据,不检查这个数据是否存在。

          c.begin()    传回迭代器中的第一个数据。

 

          c.clear()     移除容器中所有数据。

          c.empty()    判断容器是否为空。

          c.end()    指向迭代器中的最后一个数据地址。

         c.erase(pos)     删除pos位置的数据,传回下一个数据的位置。

 

         c.erase(beg,end)     删除[beg,end)区间的数据,传回下一个数据的位置。

         c.front()     传回第一个数据。

         get_allocator    使用构造函数返回一个拷贝。

         c.insert(pos,elem)     在pos位置插入一个elem拷贝,传回新数据位置

         c.insert(pos,n,elem)     在pos位置插入>n个elem数据。无返回值

         c.insert(pos,beg,end)    在pos位置插入在[beg,end)区间的数据。无返回值

         c.max_size()     返回容器中最大数据的数量。

         c.pop_back()     删除最后一个数据。

         c.pop_front()    删除头部数据。

         c.push_back(elem)    在尾部加入一个数据。

         c.push_front(elem)    在头部插入一个数据。

         c.rbegin()    传回一个逆向队列的第一个数据。

         c.rend()    传回一个逆向队列的最后一个数据的下一个位置。

 

         c.resize(num)     重新指定队列的长度。

         c.size()    返回容器中实际数据的个数。

         c.swap(c2)   将c1和c2元素互换。

         swap(c1,c2)     将c1和c2元素互换。

特点:

1、支持随机访问,即支持[]以及at(),但是性能没有vector好。

2、支持两端操作,push(pop)-back(front),但性能不及list。

最佳使用情况:

1、需要在两端插入和删除元素。

2、无需引用容器内的元素。

3、要求容器释放不再使用的元素。













本文转自朱先忠老师51CTO博客,原文链接:http://blog.51cto.com/zhuxianzhong/1546356 ,如需转载请自行联系原作者



相关文章
|
5月前
|
存储 弹性计算 人工智能
2026年阿里云计算巢部署OpenClaw(Clawdbot)小白教程
在AI自动化技术飞速普及的2026年,OpenClaw(原Clawdbot/Moltbot)作为开源的AI自动化代理工具,凭借自然语言交互、多场景任务自动化、插件化扩展的核心能力,已成为企业级自动化办公、轻量化数字化转型以及个人高效办公的核心抓手。相较于传统AI工具“只会说不会做”的局限,OpenClaw更像是一位“永不疲倦的数字员工”,能够通过简单的自然语言指令,自主完成文件处理、日程管理、邮件整理、跨平台协同等各类流程化、重复性任务,无需用户掌握复杂的编程技能,真正实现“指令下达,万事搞定”。
614 1
|
7月前
|
存储 人工智能 芯片
边缘 AI 芯片,为啥越来越“不像芯片”?聊聊这些年我看到的架构创新
边缘 AI 芯片,为啥越来越“不像芯片”?聊聊这些年我看到的架构创新
517 2
|
4月前
|
JavaScript Linux API
OpenClaw终极指南:从搭建到高阶玩法解锁(阿里云/本地部署+百炼API配置+避坑指南)
2026年,OpenClaw已从单一对话工具进化为“全场景生产力引擎”,但多数用户仍停留在基础聊天层面,未能发掘其128K超长上下文、多格式文件解析、联网搜索、代码生成等核心能力。这款工具的真正价值,在于通过灵活部署、模型适配与高阶功能组合,成为工作与学习中“不可或缺的效率伙伴”。
1275 9
|
4月前
|
人工智能 运维 监控
从零启动AI轻创业:OpenClaw全平台部署、六大实战方向与免费模型接入完整方案
2026年,开源AI智能体框架已经成为轻量化创业的核心基础设施,OpenClaw(Clawdbot)凭借可自托管、多平台接入、多Agent协同、技能化扩展等特性,为个人与小型团队提供了低门槛、高可控、可商业化的落地路径。它不再是单一的对话助手,而是集模型调度、任务执行、流程自动化、系统集成于一体的私有AI运行环境,可以在不依赖外部平台、不上传核心数据的前提下,为企业与个人提供稳定可用的AI能力。
1028 4
|
8月前
|
存储 弹性计算 人工智能
阿里云渠道商:阿里云 ecs 和轻量服务器有什么区别?
阿里云ECS与轻量应用服务器有何区别?本文详解两者在性能、成本、适用场景及扩展能力上的核心差异,结合实际案例提供选型建议,助个人开发者与企业精准匹配需求,实现高效上云。
|
11月前
|
边缘计算 运维 算法
MyEMS 开源能源管理系统:革新能源管控模式的技术实践与生态构建
在全球能源转型与“双碳”目标推动下助力重点用能单位实现高效能源管控。其模块化架构支持多协议接入、AI深度分析、数字孪生可视化,广泛适配各类场景,构建“技术开源+生态共建”的新型能源管理体系。
360 0
|
存储 监控 API
SOA简介
SOA简介
1693 1
|
JavaScript 前端开发 搜索推荐
CSR、SSR与同构渲染全方位解析
CSR、SSR与同构渲染全方位解析
539 0
|
安全 算法 物联网
SSL/TLS:互联网通信的加密基石与安全实践
**简介:** 在数字化时代,互联网每天传输海量敏感数据,网络攻击频发。SSL/TLS协议作为网络安全的基石,通过加密技术确保数据安全传输。本文解析SSL/TLS的技术架构、密码学原理、应用场景及常见误区,探讨其在未来的发展趋势,强调持续演进以应对新型威胁的重要性。 SSL/TLS不仅保障Web安全,还广泛应用于API、邮件、物联网等领域,并遵循合规标准如PCI DSS和GDPR。
|
中间件 API 开发者
深入理解Python Web框架:中间件的工作原理与应用策略
【7月更文挑战第19天】Python Web中间件摘要:**中间件是扩展框架功能的关键组件,它拦截并处理请求与响应。在Flask中,通过`before_request`和`after_request`装饰器模拟中间件行为;Django则有官方中间件系统,需实现如`process_request`和`process_response`等方法。中间件用于日志、验证等场景,但应考虑性能、执行顺序、错误处理和代码可维护性。
441 0