C++实现汉诺塔游戏用户交互

article/2025/6/10 11:58:15

目录

  • 一、模型调整
    • (一)模型定义
    • (二)模型实现
      • 1.电脑自动完成部分
      • 2.SDL图形显示
        • 2.1拿起放下盘子的函数
        • 2.2左右移动手指的函数
  • 二、处理用户输入,进行人机分流
  • 三、总结
  • 四、源码下载

上篇文章使用C++语言实现汉诺塔游戏电脑自动完成的步骤,还没有实现用户交互,无多少可玩性,本篇继续探索,加入方向键交互功能。

在这里插入图片描述

一、模型调整

(一)模型定义

这里更新的知识点包括:类的静态变量、类的构造函数。

//汉诺塔柱子,将大部分属性静态化
class Stick {public:deque<int> plates; //柱子放置盘子的栈int x, y; //柱子底中心位置static int width, height; //柱子宽、高static unsigned char r, g, b; //柱子颜色static int pside, pheight; //盘子边长、厚度static unsigned char pr, pg, pb; //盘子颜色Stick(int x, int y, int width, int height, unsigned char r, unsigned char g, unsigned char b,int pside, int pheight, unsigned char pr, unsigned char pg, unsigned char pb); //构造函数void show(SDL_Renderer*); //在指定渲染器绘制画面
};//定义静态变量
int Stick::width; int Stick::height;
unsigned char Stick::r; unsigned char Stick::g; unsigned char Stick::b;
int Stick::pside; int Stick::pheight;
unsigned char Stick::pr; unsigned char Stick::pg = pg; unsigned char Stick::pb = pb;//默认构造函数,柱子在左边沿位置,宽6,高300,棕色,盘子边长40、厚10、天蓝色
Stick::Stick(int x = 0, int y = 300, int width = 6, int height = 300, unsigned char r = 95, unsigned char g = 60, unsigned char b = 10,int pside = 40, int pheight = 10, unsigned char pr = 0, unsigned char pg = 255, unsigned char pb = 255):x(x), y(y) {Stick::width = width; Stick::height = height;Stick::r = r; Stick::g = g; Stick::b = b;Stick::pside = pside; Stick::pheight = pheight;Stick::pr = pr; Stick::pg = pg; Stick::pb = pb;}//整个汉诺塔模型,新增手指、空中盘子2个变量,新增upPlate、downPlate、leftFiger、rightFinger4个成员函数
//更新了显示函数,使用户可以设置刷新频率
class Hanoi {private:SDL_Renderer * render; // 内置窗口渲染器指针Finger finger; //手指int airPlate; //空中的盘子号public:Stick sticks[3]; //3根柱子Hanoi(SDL_Renderer *, int, int); //构造函数bool movePlate(int a, int b); //将1个盘子从a柱移到b柱void movePlates(int a, int b, int count); //将count个盘子从a柱移到b柱bool upPlate(); //从手指处拾起1个盘子bool downPlate(); //在手指处放下1个盘子void leftFinger(); //手指左移1下void rightFinger(); //手指右移1下void show(); //在窗口绘制画面void show(int); //在窗口以指定时间间隔绘制画面
};//手指颜色默认未拿起时为蓝色,在第0根柱子上
Hanoi::Hanoi(SDL_Renderer *render, int n, int index=0): render(render), finger({index, 0, 0, 255}) {//初始化空中无盘子airPlate = 0;//初始化第0根柱子,盘子从n到1压栈for (int i = n; i >0; i--) {sticks[0].plates.push_back(i);}//明确3根柱子位置sticks[0].x = 200; sticks[1].x = 400, sticks[2].x = 600;
}//手指模型
typedef struct Finger {int index; //柱号unsigned char r, g, b; //手指颜色
} Finger;

(二)模型实现

1.电脑自动完成部分

这里更新的知识点包括:移动圆盘前后SDL显示、手指状态显示

bool Hanoi::movePlate(int a, int b) {int plateA, plateB; //2根柱子顶端的盘子plateA = this->sticks[a].plates.back();if (!this->sticks[b].plates.empty()) {plateB = this->sticks[b].plates.back();if (plateA > plateB) return false;}//小盘子可以放在大盘子上this->finger.index = a;this->finger.r = 255; this->finger.b = 0; //让手指变红成拿起圆盘状态this->show();this->sticks[a].plates.pop_back();this->sticks[b].plates.push_back(plateA);this->finger.index = b;this->finger.r = 0; this->finger.b = 255; //让手指变蓝成旆圆盘状态this->show(); //显示return true;
}

2.SDL图形显示

这里更新的知识点包括:SDL事件捕获与响应

SDL_Event event;
while(true) {SDL_PollEvent(&event); //等待事件
// cout << event.type << " " << flush;if (event.type == SDL_QUIT) { // 退出break;} else if (event.type == SDL_KEYDOWN) { // 按键//防抖while(true) {SDL_PollEvent(&event); //等待事件if ((event.type == SDL_KEYUP)) break;}switch (event.key.keysym.sym) { //检测按了哪个方向键case SDLK_UP:hanoi.upPlate();break;case SDLK_DOWN:hanoi.downPlate();break;case SDLK_LEFT:hanoi.leftFinger();break;case SDLK_RIGHT:hanoi.rightFinger();break;}}hanoi.show(20); //20毫秒刷新一次
2.1拿起放下盘子的函数
bool Hanoi::upPlate() {if (this->sticks[this->finger.index].plates.empty() || this->airPlate) return false; //柱子为空或已经有空中盘子// 弹出圆盘this->airPlate = this->sticks[this->finger.index].plates.back();this->sticks[this->finger.index].plates.pop_back();// 手指变红this->finger.r = 255; this->finger.b = 0;return true;
}bool Hanoi::downPlate() {//空中无圆盘或下方圆盘小都不能放if (this->airPlate == 0) return false; if (this->sticks[this->finger.index].plates.size() > 0 && this->sticks[this->finger.index].plates.back() < this->airPlate)return false;// 压入圆盘this->sticks[this->finger.index].plates.push_back(this->airPlate);this->airPlate = 0;// 手指变蓝this->finger.r = 0; this->finger.b = 255;return true;
}
2.2左右移动手指的函数
void Hanoi::leftFinger() {this->finger.index = (this->finger.index + 2) % 3;
}void Hanoi::rightFinger() {this->finger.index = (this->finger.index + 1) % 3;
}

二、处理用户输入,进行人机分流

放置在main函数中:

    int n; //汉诺塔层数int mode; //游戏模式while (true) {cout << "请输入汉诺塔层数:" << flush;cin >> n;cout << "选择模式:1:电脑完成,2:玩家完成" << endl;cin >> mode;if (mode != 1 && mode !=2) {cout << "模式错误" << endl;} else {//初始化SDL部分//...//初始化汉诺塔部分Hanoi hanoi(render, n);hanoi.show(); //修复刚启动黑屏bugif (mode == 1) { //电脑自动完成部分hanoi.movePlates(0, 2, n);} else { //mode == 2,用户交互部分//进行事件捕获与响应//...}//SDL销毁部分//...}}

三、总结

本篇文章摘记了使用C++语言实现汉诺塔游戏电脑自动完成和用户交互功能的步骤,希望大家有所收获,如有好的建议欢迎留言,谢谢大家!

四、源码下载

C++实现图形化汉诺塔游戏源代码


http://www.hkcw.cn/article/aqFPLLGJml.shtml

相关文章

嵌入式学习 D32:系统编程--进程间通信IPC

引言--进程间通信管道的概念管道相关操作有名管道及其相关操作信号通信 一、引言--进程间通信 1&#xff09;因为空间是独立和隔绝的&#xff0c;数据发不过去&#xff0c;需要进程间的通信来交互&#xff0c;所以需要通信。 2&#xff09;linux进程间通信的常用几种方式&…

黑马Java面试笔记之 消息中间件篇(Kafka)

一. Kafka保证消息不丢失 Kafka如何保证消息不丢失 使用Kafka在消息的收发过程中都会出现消息丢失&#xff0c;Kafka分别给出了解决方案 生产者发送消息到Brocker丢失消息在Brocker中存储丢失消费者从Brocker接收消息丢失 1.1 生产者发送消息到Brocker丢失 设置异步发送 消息…

java的SPI机制

SPI&#xff08;Service Provider Interface&#xff09;是java提供的一种服务发现机制。允许你定义一个接口或抽象类&#xff0c;然后由第三方实现这个接口&#xff0c;并在运行时动态加载这些实现类 核心思想是&#xff1a;面向接口编程&#xff0c;解耦接口与实现 核心组件…

SpringCloud 分布式锁Redisson锁的重入性 高并发 获取锁

介绍 Redisson 的锁支持 可重入性&#xff0c;这意味着同一个线程在获取锁后&#xff0c;如果再次尝试获取该锁&#xff0c;它可以成功地获得锁&#xff0c;而不会被阻塞。 每次一个线程成功获取锁后&#xff0c;它的持有次数会增加。当线程再次获取该锁时&#xff0c;Rediss…

PyTorch--池化层(4)

池化层&#xff08;Pooling Layer&#xff09; 用于降低特征图的空间维度&#xff0c;减少计算量和参数数量&#xff0c;同时保留最重要的特征信息。 池化作用&#xff1a;比如1080p视频——720p 池化层的步长默认是卷积核的大小 ceil 允许有出界部分&#xff1b;floor 不允许…

【自动思考记忆系统】demo (Java版)

背景&#xff1a;看了《人工智能》中的一段文章&#xff0c;于是有了想法。想从另一种观点&#xff08;⭕️&#xff09;出发&#xff0c;尝试编码&#xff0c;告别传统程序员一段代码解决一个问题的方式。下图是文章原文和我的思考涂鸦✍️&#xff0c;于是想写一个自动思考记…

小白的进阶之路系列之十二----人工智能从初步到精通pytorch综合运用的讲解第五部分

在本笔记本中,我们将针对Fashion-MNIST数据集训练LeNet-5的变体。Fashion-MNIST是一组描绘各种服装的图像瓦片,有十个类别标签表明所描绘的服装类型。 # PyTorch model and training necessities import torch import torch.nn as nn import torch.nn.functional as F impor…

pytorch3d+pytorch1.10+MinkowskiEngine安装

1、配置pytorch1.10cuda11.0 pip install torch1.10.1cu111 torchvision0.11.2cu111 torchaudio0.10.1 -f https://download.pytorch.org/whl/cu111/torch_stable.html 2、配置 MinkowskiEngine库 不按下面步骤&#xff0c;出现错误 1、下载MinkowskiEngine0.5.4到本地 2、查看…

ORACLE 缺失 OracleDBConsoleorcl服务导致https://xxx:port/em 不能访问

这个原因是&#xff0c;操作过一下 ORCL的服务配置变更导致的。 再PATH中添加个环境变量&#xff0c;路径如下 管理员权限运行cmd 等待创建完成 大概3分钟 查看服务 点击第一个访问&#xff0c;下图登录后的截图

分布式流处理与消息传递——向量时钟 (Vector Clocks) 算法详解

Java 实现向量时钟 (Vector Clocks) 算法详解 一、向量时钟核心原理 #mermaid-svg-JcZ1GT0r1ZNSy6W7 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-JcZ1GT0r1ZNSy6W7 .error-icon{fill:#552222;}#mermaid-svg-JcZ…

深入浅出:Oracle 数据库 SQL 执行计划查看详解(1)——基础概念与查看方式

背景 在当今的软件开发领域&#xff0c;尽管主流开发模式往往倾向于采用单表模式&#xff0c;力图尽可能地减少表之间的连接操作&#xff0c;以期达到提高数据处理效率、简化应用逻辑等目的。然而&#xff0c;对于那些已经上线运行多年的运维老系统而言&#xff0c;它们内部往…

多模态大语言模型arxiv论文略读(104)

Talk Less, Interact Better: Evaluating In-context Conversational Adaptation in Multimodal LLMs ➡️ 论文标题&#xff1a;Talk Less, Interact Better: Evaluating In-context Conversational Adaptation in Multimodal LLMs ➡️ 论文作者&#xff1a;Yilun Hua, Yoav…

【Oracle】游标

个人主页&#xff1a;Guiat 归属专栏&#xff1a;Oracle 文章目录 1. 游标基础概述1.1 游标的概念与作用1.2 游标的生命周期1.3 游标的分类 2. 显式游标2.1 显式游标的基本语法2.1.1 声明游标2.1.2 带参数的游标 2.2 游标的基本操作2.2.1 完整的游标操作示例 2.3 游标属性2.3.1…

Ethernet/IP转DeviceNet网关:驱动大型矿山自动化升级的核心纽带

在大型矿山自动化系统中&#xff0c;如何高效整合新老设备、打通数据孤岛、实现统一控制&#xff0c;是提升效率与安全的关键挑战。JH-EIP-DVN疆鸿智能EtherNet/IP转DeviceNet网关&#xff0c;正是解决这一难题的核心桥梁&#xff0c;为矿山各环节注入强劲连接力&#xff1a; …

Nginx + Tomcat 负载均衡、动静分离群集

一、 nginx 简介 Nginx 是一款轻量级的高性能 Web 服务器、反向代理服务器及电子邮件&#xff08;IMAP/POP3&#xff09;代理服务器&#xff0c;在 BSD-like 协议下发行。其特点是占有内存少&#xff0c;并发能力强&#xff0c;在同类型的网页服务器中表现优异&#xff0c;常用…

5.Nginx+Tomcat负载均衡群集

Tomcat服务器应用场景&#xff1a;tomcat服务器是一个免费的开放源代码的Web应用服务器&#xff0c;属于轻量级应用服务器&#xff0c;在中小型系统和并发访问用户不是很多的场合下被普遍使用&#xff0c;是开发和调试JSP程序的首选。一般来说&#xff0c;Tomcat虽然和Apache或…

【算法设计与分析】实验——汽车加油问题, 删数问题(算法实现:代码,测试用例,结果分析,算法思路分析,总结)

说明&#xff1a;博主是大学生&#xff0c;有一门课是算法设计与分析&#xff0c;这是博主记录课程实验报告的内容&#xff0c;题目是老师给的&#xff0c;其他内容和代码均为原创&#xff0c;可以参考学习&#xff0c;转载和搬运需评论吱声并注明出处哦。 4-1算法实现题 汽车…

网络爬虫 - App爬虫及代理的使用(十一)

App爬虫及代理的使用 一、App抓包1. App爬虫原理2. reqable的安装与配置1. reqable安装教程2. reqable的配置3. 模拟器的安装与配置1. 夜神模拟器的安装2. 夜神模拟器的配置4. 内联调试及注意事项1. 软件启动顺序2. 开启抓包功能3. reqable面板功能4. 夜神模拟器设置项5. 注意事…

SQLite详细解读

一、SQLite 是什么&#xff1f; SQLite 是一个嵌入式关系型数据库管理系统&#xff08;RDBMS&#xff09;。它不是像 MySQL 或 PostgreSQL 那样的客户端-服务器数据库引擎&#xff0c;而是一个自包含的、无服务器的、零配置的、事务性的 SQL 数据库引擎。 核心特点 嵌入式/库…

线程池详细解析(三)

本章我们来讲一讲线程池的最后一个方法shutdown&#xff0c;这个方法的主要作用就是将线程池进行关闭 shutdown&#xff1a; public void shutdown() {ReentrantLock var1 this.mainLock;var1.lock();try {this.checkShutdownAccess();this.advanceRunState(0);this.interrup…