C++实现俄罗斯方块
俄罗斯方块
还记得俄罗斯方块吗?相信这是小时候我们每个人都喜欢玩的一个小游戏。顾名思义,俄罗斯方块自然是俄罗斯人发明的。这人叫阿列克谢·帕基特诺夫。他设置这个游戏的规则是:由小方块组成的不同形状的板块陆续从屏幕上方落下来,玩家通过调整板块的位置和方向,使它们在屏幕底部拼出完整的一条或几条。这些完整的横条会随即消失,给新落下来的板块腾出空间,与此同时,玩家得到分数奖励。没有被消除掉的方块不断堆积起来,一旦堆到屏幕顶端,玩家便告输,游戏结束。

实现思路
那么问题就来了,如何用我们学过的C++去实现这个游戏呢?其实在写代码的时候,我们遵循的一个策略就是“拆分法”,从大的框架去一步步拆解成每个小的部分,然后这每个小的部分你都能用C++去实现它;要是拆分的小部分你还是实现不了,那就继续拆分,知道你能实现为止。比如这个俄罗斯方块的游戏,你的思路应该是这样:
1、画出游戏地图, 并留出下一图形和分数显示的位置
2、图形的建立和颜色
3、图形下落的实现以及上一图形的清除
4、是否能继续下落或变形的检测
5、某一行是否已满需清除以及清除功能与分数更新
实现代码如下:
#include<iostream>
#include<string>
#include<cstdlib>
#include<windows.h>
#include<ctime>
#include<conio.h>
#include<cstdio>
using namespace std;class Tetris
{
private:int rank; //游戏难度等级int score; // 得分int id; //图形IDint point[2]; //两基点int top; //最高点高度
public:Tetris();void Welocme(); //首界面void DrawMap(); //游戏界面void SetColor(int); //控制颜色void Draw(int, int, int); //画图形void Run(); //运行游戏void ReDraw(int, int, int); //清除图形bool Judge(int, int, int);void Turn(int); //旋转void Updata(); // 更新界面void Pause(); //游戏暂停void Input_score();
};const int sharp[15][8] = //组成图形的各个点的各个坐标,先纵后横
{
{0,0,1,0,2,0,3,0},{0,0,0,1,0,2,0,3},
{0,0,1,0,0,1,1,1},
{0,0,1,0,1,1,1,2},{0,1,1,1,2,0,2,1},{0,0,0,1,0,2,1,2},{0,0,0,1,1,0,2,0},
{1,0,1,1,1,2,0,2},{0,0,0,1,1,1,2,1},{0,0,0,1,0,2,1,0},{0,0,1,0,2,0,2,1},
{0,0,0,1,1,1,1,2},{0,1,1,0,1,1,2,0},
{0,1,0,2,1,0,1,1},{0,0,1,0,1,1,2,1}
};const int high[15] = { 4,1,2,2,3,2,3,2,3,2,3,2,3,2,3 };
int map[28][16];#define a1 0 //条形
#define a2 1
#define b 2 // 方块#define c1 3 //L形
#define c2 4
#define c3 5
#define c4 6#define d1 7 //T形
#define d2 8
#define d3 9
#define d4 10#define e1 11 //闪电1形
#define e2 12#define f1 13 //闪电2形
#define f2 14Tetris::Tetris() //构造函数, 初始化各个值
{point[0] = 0;point[1] = 5;score = 0;top = 25;
}void Tetris::Turn(int num) //旋转函数
{switch (num){case a1: id = a2; break; //条形互换case a2: id = a1; break;case b: id = b; break; //方块无法旋转case c1: id = c2; break; //各种L形互换case c2: id = c3; break;case c3: id = c4; break;case c4: id = c1; break;case d1: id = d2; break; //各种T形互换case d2: id = d3; break;case d3: id = d4; break;case d4: id = d1; break;case e1: id = e2; break; //两种闪电形互换case e2: id = e1; break;case f1: id = f2; break;case f2: id = f1; break;}
}void SetPos(int i, int j) //控制光标位置, 列, 行
{COORD pos = { i,j };SetConsoleCursorPosition(GetStdHandle(STD_OUTPUT_HANDLE), pos);
}void Tetris::Pause() // 暂停函数
{SetPos(32, 10);cout << "游戏暂停!" << endl;SetPos(30, 11);cout << "你的分数为 " << score;char temp;while (1){while (1){if (_kbhit()){temp = _getch();break;}}if (temp == 32)break;}SetPos(32, 10); // 清除暂停时显示的信息cout << " ";SetPos(30, 11);cout << " ";
}void Tetris::Updata() //更新函数
{int i, flag;int nx, ny;for (i = 0; i < 4; i++){nx = point[0] + sharp[id][i * 2];ny = point[1] + sharp[id][i * 2 + 1];SetPos((ny + 1) * 2, nx + 1);SetColor(0);cout << "■";map[nx][ny] = 1; //界面各个点是否为空的更新}if (point[0] < top)top = point[0]; //最高点的更新for (i = point[0]; i < point[0] + high[id]; i++) //消除行{flag = 1;for (int j = 0; j < 13; j++) //判定某一行是否满, 用flag来标记if (map[i][j] == 0)flag = 0;if (flag == 1){for (int k = i; k >= top; k--){for (int p = 0; p < 13; p++){map[k][p] = map[k - 1][p];SetPos((p + 1) * 2, k + 1);if (map[k][p] == 1)cout << "■";else cout << " ";}}score += 10;Input_score();}}
}void Tetris::Input_score()
{SetColor(3);SetPos(30, 19);cout << "得分: " << score;
}void Tetris::Welocme() //欢迎界面
{SetColor(1);char x;while (1){system("cls");cout << "■■■■■■■■■■■■■■■■■■■■■" << endl;cout << " 俄罗斯方块 " << endl;cout << "■■■■■■■■■■■■■■■■■■■■■" << endl;cout << " 操作方式:" << endl;cout << " ↑ - 旋转" << endl;cout << " ↓ - 加速下移" << endl;cout << " ← - 左移" << endl;cout << " → - 右移" << endl;cout << " 空格 - 暂停" << endl;cout << "■■■■■■■■■■■■■■■■■■■■■" << endl;cout << "■ 按1—3选择难度■" << endl;SetPos(20, 10);x = getchar();if (x <= '9' && x >= '0'){rank = x - '0';break;}}
}void Tetris::SetColor(int color_num) //设置颜色
{int n;switch (color_num){case 0: n = 0x08; break;case 1: n = 0x0C; break;case 2: n = 0x0D; break;case 3: n = 0x0E; break;case 4: n = 0x0A; break;}SetConsoleTextAttribute(GetStdHandle(STD_OUTPUT_HANDLE), n);
}void Tetris::DrawMap() //画游戏时界面
{int i;SetColor(0);for (i = 0; i < 24; i++) //宽24格{SetPos(i * 2, 0);cout << "■";SetPos(i * 2, 26);cout << "■";}for (i = 0; i < 26; i++) //高26格{SetPos(0, i);cout << "■";SetPos(28, i);cout << "■";SetPos(46, i);cout << "■";}for (i = 14; i < 24; i++){SetPos(i * 2, 16);cout << "■";}SetColor(3);Input_score();SetPos(30, 21);cout << "难度等级: " << rank;SetPos(32, 2);cout << "下一图形";
}void Tetris::Draw(int x, int y, int num) //画图形
{int nx, ny;for (int i = 0; i < 4; i++){nx = x + sharp[num][2 * i];ny = y + sharp[num][2 * i + 1];SetPos((ny + 1) * 2, nx + 1);SetColor(i + 1);cout << "■";}
}void Tetris::ReDraw(int x, int y, int num) //为更新图形的位置清除图形
{int nx, ny;for (int i = 0; i < 4; i++){nx = x + sharp[num][2 * i];ny = y + sharp[num][2 * i + 1];SetPos((ny + 1) * 2, nx + 1);cout << " ";}
}bool Tetris::Judge(int x, int y, int num) //判定在x, y 所指位置是否可画编号为
{ //num 的图形, 若不可画则反回trueint nx, ny;for (int i = 0; i < 4; i++){nx = x + sharp[num][2 * i];ny = y + sharp[num][2 * i + 1];if (!(nx < 25 && nx >= 0 && ny < 13 && ny >= 0 && !map[nx][ny]))return true;}return false;
}void Tetris::Run() //运行游戏
{int next_id;srand((int)time(0));id = rand() % 15;next_id = rand() % 15;Draw(point[0], point[1], id);Draw(5, 16, next_id);int count;if (rank == 1)count = 150;else if (rank == 2)count = 100;else if (rank==3)count = 50;elsecount = 5;int i = 0; //不同等级对应不同countwhile (1){if (!(i < count)) //i 与 count 用于控制时间{i = 0;if (Judge(point[0] + 1, point[1], id)) //在某一位置不能下落的话{Updata();id = next_id;ReDraw(5, 16, next_id);next_id = rand() % 15;point[0] = 0; point[1] = 5;Draw(point[0], point[1], id);Draw(5, 16, next_id);if (Judge(point[0], point[1], id)){system("cls");SetPos(20, 10);cout << "游戏结束!" << endl;SetPos(20, 11);cout << "你的分数为 " << score << endl;system("pause");exit(1);}}else //继续下落{ReDraw(point[0], point[1], id);point[0]++;Draw(point[0], point[1], id);}}if (_kbhit()) //键盘输入值时 {int key, key2;key = _getch();if (key == 224){key2 = _getch();if (key2 == 72) //按向上方向键时{int temp = id;Turn(id);if (Judge(point[0], point[1], id))id = temp;ReDraw(point[0], point[1], temp);Draw(point[0], point[1], id);}if (key2 == 80) //按向下方向键时{if (!Judge(point[0] + 2, point[1], id)){ReDraw(point[0], point[1], id);point[0] += 2;Draw(point[0], point[1], id);}}else if (key2 == 75) //按向左方向键时{if (!Judge(point[0], point[1] - 1, id)){ReDraw(point[0], point[1], id);point[1]--;Draw(point[0], point[1], id);}}else if (key2 == 77) //按向右方向键时{if (!Judge(point[0], point[1] + 1, id)){ReDraw(point[0], point[1], id);point[1]++;Draw(point[0], point[1], id);}}}else if (key == 32) // 按下空格暂停Pause();}Sleep(1); //等待1毫秒i++; //控制下落间隔}
}int main()
{Tetris game;game.Welocme();system("cls"); //清除欢迎界面game.DrawMap();game.Run();
}
运行效果如下图所示:

相关文章:
C++实现俄罗斯方块
俄罗斯方块 还记得俄罗斯方块吗?相信这是小时候我们每个人都喜欢玩的一个小游戏。顾名思义,俄罗斯方块自然是俄罗斯人发明的。这人叫阿列克谢帕基特诺夫。他设置这个游戏的规则是:由小方块组成的不同形状的板块陆续从屏幕上方落下来…...
鸿蒙分享:添加模块,修改app名称图标
新建公共模块common 在entry的oh-package.json5添加dependencies,引入common模块 "dependencies": {"common": "file:../common" } 修改app名称: common--src--resources--string.json 新增: {"name&q…...
扫描IP段内的使用的IP
扫描IP段内的使用的IP 方法一:命令行 命令行进入 for /L %i IN (1,1,254) DO ping -w 1 -n 1 192.168.3.%iarp -a方法二:python from scapy.all import ARP, Ether, srp import keyboarddef scan_network(ip_range):# 创建一个ARP请求包arp ARP(pds…...
【专题】虚拟存储器
前文提到的存储器管理方式有一个共同的特点,即它们都要求将一个作业全部装入内存后方能运行。 但有两种特殊情况: 有的作业很大,其所要求的内存空间超过了内存总容量,作业不能全部被装入内存,致使该作业无法运行&#…...
Python之爬虫入门--示例(2)
一、Requests库安装 可以使用命令提示符指令直接安装requests库使用 pip install requests 二、爬取JSON数据 (1)、点击网络 (2)、刷新网页 (3)、这里有一些数据类型,选择全部 (…...
5G CPE终端功能及性能评测(四)
5G CPE 功能性能评测 本文选取了几款在工业应用领域应用较多的5G CPE,对其功能和性能进行了对比评测。功能方面主要对比了网络接口数量,VPN功能 支持情况。以下测试为空口测试,测试结果受环境影响较大,性能仅供参考。总体看,高通X55芯片下行最优,速率稳定。 功能 对比CPE…...
人工智能驱动的骗局会模仿熟悉的声音
由于人工智能技术的进步,各种现代骗局变得越来越复杂。 这些骗局现在包括人工智能驱动的网络钓鱼技术,即使用人工智能模仿家人或朋友的声音和视频。 诈骗者使用来自社交媒体的内容来制作深度伪造内容,要求提供金钱或个人信息。个人应该通过…...
电子病历静态数据脱敏路径探索
一、引言 数据脱敏(Data Masking),屏蔽敏感数据,对某些敏感信息(比如patient_name、ip_no、ad、no、icd11、drug等等 )通过脱敏规则进行数据的变形,实现隐私数据的可靠保护。电子病历作为医疗领…...
混合云策略在安全领域受到青睐
Genetec 发布了《2025 年物理安全状况报告》,该报告根据超过 5,600 名该领域领导者(其中包括 100 多名来自澳大利亚和新西兰的领导者)的回应,揭示了物理安全运营的趋势。 报告发现,澳大利亚和新西兰的组织采用混合云策…...
Echarts使用平面方法绘制三维立体柱状图表
目录 一、准备工作 1.下载引入ECharts库 2.创建容器 二、绘制基本柱状 三、绘制立体柱状方法一 1.定义立方体形状 2.注册立方体形状 3.配置custom系列 4.设置数据 5.渲染图表 四、绘制立体柱状方法二 1.画前知识 2.计算坐标renderItem 函数 (1&#x…...
java-判断语句
题目一:选择练习1 657. 选择练习1 - AcWing题库 代码 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);int a sc.nextInt(), b sc.nextInt();int c sc.nextInt(), d sc.nextInt();…...
11.14【JAVA EXP3】【DEBUG】
比较疑惑的一点是当前页面(资源的url)与请求的url? 请求的url由webService接收,servelt当中也可以发送出这个url 进行页面跳转,是跳转到某个Jsp页面,这个页面的url是在哪里定义的? 在Jsp打印信息,这个报…...
UE5 和 UE4 中常用的控制台命令总结
调用控制台 按下键盘上的 ~ 键可以调用控制台命令。 技巧 使用键盘的 ↑ 键可以查看之前输入过的指令。控制台指令并不需要打全名,输入空格后跟随指令的部分字符可以进行模糊搜索。按下 Ctrl Shift , 打开 GPUProfile 面板。 命令如下: 调试类 s…...
MR30分布式IO模块赋能喷水织机
纺织行业作为我国传统支柱产业,历经数千年的演变,如今仍面临着诸多困境,在纺织行业中,每一次技术的飞跃都是对行业边界的勇敢探索。在纺织行业,喷水织机作为关键生产设备,其性能直接影响到产品质量和产能。…...
C++中的封装性
定义: 封装性: 1.将属性(成员变量)和行为(成员函数)作为一个整体,表现在生活中的事物 2.将属性和行为加以权限控制 (将事物的属性(成员变量)和行为&#…...
PyTorch 深度学习框架简介:灵活、高效的 AI 开发工具
PyTorch 深度学习框架简介:灵活、高效的 AI 开发工具 PyTorch 作为一个深度学习框架,以其灵活性、可扩展性和高效性广受欢迎。无论是在研究领域进行创新实验,还是在工业界构建生产级的深度学习模型,PyTorch 都能提供所需的工具和…...
leetcode-22.括号生成
暴力 感谢分享这个思路和算法。生成括号的问题可以通过生成所有可能的括号序列并验证其有效性来解决。以下是对该思路的详细解释和实现: 思路 生成所有可能的序列: 使用递归生成所有长度为 2n 的括号序列。在每个位置可以选择放置 ( 或 )。 验证序列的…...
devops-Dockerfile+Jenkinsfile方式部署Java前后端应用
文章目录 概述部署前端Vue应用一、环境准备1、Dockerfile2、.dockerignore3、nginx.conf4、Jenkinsfile 二、Jenkins部署1、新建任务2、流水线3、Build Now 构建 & 访问 Springboot后端应用1. 准备工作2. 创建项目结构3. 编写 Dockerfile后端 Dockerfile (backend/Dockerfi…...
【Apache Paimon】-- 4 -- Flink 消费 kafka 数据,然后写入 paimon
目录 1、本地开发环境 2、kafka2paimon 实现流程 3、代码实现 3.1、项目名称 3.2、项目结构 3.3、Pom.xml 和 log4j.properties 文件 3.4、代码核心类 3.4.1、入口类:Kafka2PaimonDemo.java 3.4.2、参数解析类 3.4.2.1、JobParameterUtil.java( flink job schedule…...
【成功解决】:VS2019(Visual Studio 2019)遇到E2870问题:此配置中不支持 128 位浮点类型
起因:项目中需要用json来操作数据,就引了cJSON库(cJSON.h和cJSON.c文件),但是发现编译报错如下 E2870 此配置中不支持 128 位浮点类型 test0 ...\usr\include\x86_64-linux-gnu\bits\floatn.h 75 然后先新建了个工程来检查问题(甚至在这之前还以为是cjson…...
浅谈 React Hooks
React Hooks 是 React 16.8 引入的一组 API,用于在函数组件中使用 state 和其他 React 特性(例如生命周期方法、context 等)。Hooks 通过简洁的函数接口,解决了状态与 UI 的高度解耦,通过函数式编程范式实现更灵活 Rea…...
基于大模型的 UI 自动化系统
基于大模型的 UI 自动化系统 下面是一个完整的 Python 系统,利用大模型实现智能 UI 自动化,结合计算机视觉和自然语言处理技术,实现"看屏操作"的能力。 系统架构设计 #mermaid-svg-2gn2GRvh5WCP2ktF {font-family:"trebuchet ms",verdana,arial,sans-…...
2025年能源电力系统与流体力学国际会议 (EPSFD 2025)
2025年能源电力系统与流体力学国际会议(EPSFD 2025)将于本年度在美丽的杭州盛大召开。作为全球能源、电力系统以及流体力学领域的顶级盛会,EPSFD 2025旨在为来自世界各地的科学家、工程师和研究人员提供一个展示最新研究成果、分享实践经验及…...
基于uniapp+WebSocket实现聊天对话、消息监听、消息推送、聊天室等功能,多端兼容
基于 UniApp + WebSocket实现多端兼容的实时通讯系统,涵盖WebSocket连接建立、消息收发机制、多端兼容性配置、消息实时监听等功能,适配微信小程序、H5、Android、iOS等终端 目录 技术选型分析WebSocket协议优势UniApp跨平台特性WebSocket 基础实现连接管理消息收发连接…...
STM32+rt-thread判断是否联网
一、根据NETDEV_FLAG_INTERNET_UP位判断 static bool is_conncected(void) {struct netdev *dev RT_NULL;dev netdev_get_first_by_flags(NETDEV_FLAG_INTERNET_UP);if (dev RT_NULL){printf("wait netdev internet up...");return false;}else{printf("loc…...
UE5 学习系列(三)创建和移动物体
这篇博客是该系列的第三篇,是在之前两篇博客的基础上展开,主要介绍如何在操作界面中创建和拖动物体,这篇博客跟随的视频链接如下: B 站视频:s03-创建和移动物体 如果你不打算开之前的博客并且对UE5 比较熟的话按照以…...
ESP32读取DHT11温湿度数据
芯片:ESP32 环境:Arduino 一、安装DHT11传感器库 红框的库,别安装错了 二、代码 注意,DATA口要连接在D15上 #include "DHT.h" // 包含DHT库#define DHTPIN 15 // 定义DHT11数据引脚连接到ESP32的GPIO15 #define D…...
【项目实战】通过多模态+LangGraph实现PPT生成助手
PPT自动生成系统 基于LangGraph的PPT自动生成系统,可以将Markdown文档自动转换为PPT演示文稿。 功能特点 Markdown解析:自动解析Markdown文档结构PPT模板分析:分析PPT模板的布局和风格智能布局决策:匹配内容与合适的PPT布局自动…...
PL0语法,分析器实现!
简介 PL/0 是一种简单的编程语言,通常用于教学编译原理。它的语法结构清晰,功能包括常量定义、变量声明、过程(子程序)定义以及基本的控制结构(如条件语句和循环语句)。 PL/0 语法规范 PL/0 是一种教学用的小型编程语言,由 Niklaus Wirth 设计,用于展示编译原理的核…...
BCS 2025|百度副总裁陈洋:智能体在安全领域的应用实践
6月5日,2025全球数字经济大会数字安全主论坛暨北京网络安全大会在国家会议中心隆重开幕。百度副总裁陈洋受邀出席,并作《智能体在安全领域的应用实践》主题演讲,分享了在智能体在安全领域的突破性实践。他指出,百度通过将安全能力…...
