Java实现简单的俄罗斯方块游戏

一、创建新项目

1.首先新建一个项目,并命名为俄罗斯方块。

2.其次新建一个类,命名为Main,或其他的。

二、运行代码

代码如下:

package 俄罗斯方块;
 
import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.GridLayout;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
 
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTextArea;
import javax.swing.JTextField;
 
public class Main extends JFrame implements KeyListener {
	private JTextArea[][] grids;// 把整个界面变为一个文本区域,整个游戏在里面进行
	private int data[][]; // 对于每个格子的数据,1代表有方块,0代表为空白区
	private int[] allRect; // 所有的方块类型,用16个字节来存储,俄罗斯方块图形都是在4*4格子里
	private int rect; // 当前游戏下落的方块类型;
	private int x, y; // 当前方块的坐标位置,x代表行,y代表列
	private int score = 0; // 记录当前游戏得分情况,每消一层得10分
	private JLabel label; // 显示分数的标签
	private JLabel label1;// 显示游戏是否结束
	private boolean running; // 用于判断游戏是否结束
	/*无参构造函数*/
	public Main() {
		grids = new JTextArea[26][12];//设置游戏区域行和列
		data = new int[26][12];//开辟data数组空间与游戏区域行和列一致
		allRect = new int[] { 0x00cc, 0x8888, 0x000f, 0x0c44, 0x002e, 0x088c, 0x00e8, 0x0c88, 0x00e2, 0x044c, 0x008e,
				0x08c4, 0x006c, 0x04c8, 0x00c6, 0x08c8, 0x004e, 0x04c4, 0x00e4 };//19种方块形状,如0x00cc就是   0000 表示一个2*2的正方形方块
																										//0000 
																										//1100 
																										//1100
		label = new JLabel("score: 0"); //此标签存放得分情况,初始化为0分
		label1 = new JLabel("开始游戏"); //此标签为提示游戏状态:开始还是结束
		running = false; //为标志变量,false为游戏结束,true为游戏正在进行
		init(); // 游戏界面初始化
	}
	/*游戏界面初始化函数*/
	public void init() {
		JPanel center = new JPanel(); //此面板为游戏核心区域
		JPanel right = new JPanel(); //此面板为游戏说明区域
		center.setLayout(new GridLayout(26, 12, 1, 1)); //给游戏核心区域划分行、列共26行,12列
		for (int i = 0; i < grids.length; i++) {//初始化面板
			for (int j = 0; j < grids[i].length; j++) {
				grids[i][j] = new JTextArea(20, 20);
				grids[i][j].setBackground(Color.WHITE);
				grids[i][j].addKeyListener(this);// 添加键盘监听事件
				//初始化游戏边界
				if (j == 0 || j == grids[i].length - 1 || i == grids.length - 1) {
					grids[i][j].setBackground(Color.PINK);
					data[i][j] = 1;
				}
				grids[i][j].setEditable(false);// 文本区域不可编辑
				center.add(grids[i][j]); //把文本区域添加到主面板上
			}
		}
		//初始化游戏说明面板
		right.setLayout(new GridLayout(4, 1));
		right.add(new JLabel(" a : left        d : right"));
		right.add(new JLabel(" s : down   w : change"));
		right.add(label);
		label1.setForeground(Color.RED);// 设置标签内容为红色字体
		right.add(label1);
		//把主面板和说明面板添加到窗体中
		this.setLayout(new BorderLayout());
		this.add(center, BorderLayout.CENTER);
		this.add(right, BorderLayout.EAST);
		running = true; //初始化running状态为true,表示程序运行即游戏开始
		this.setSize(600, 850);// 设置窗体大小
		this.setVisible(true);// 窗体可见
		this.setLocationRelativeTo(null);// 设置窗体居中
		this.setResizable(false);// 窗体大小不可改变
		this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);// 释放窗体
	}
	/*主函数*/
	public static void main(String[] args) {
		Main m = new Main(); //创建Main对象,主要用于初始化数据
		m.go();// 开始游戏
	}
	/*开始游戏*/
	public void go() {// 开始游戏
		while (true) {//游戏开始直到游戏失败才结束,否则一直执行
			if (running == false) {//如果游戏失败
				break;
			}
			ranRect();// 绘制下落方格形状
			start();// 开始游戏
		}
		label1.setText("游戏结束!");//则游戏结束
	}
	/*绘制下落方格形状*/
	public void ranRect() {
		rect = allRect[(int) (Math.random() * 19)];// 随机生成方块类型(共7种,19个形状)
	}
	/*游戏开始函数*/
	public void start() {
		x = 0;
		y = 5; //初始化下落方块的位置
		for (int i = 0; i < 26; i++) {//共26层,一层一层下落
			try {
				Thread.sleep(1000);//每层延时1秒
				if (canFall(x, y) == false) {// 如果不可以掉落
					saveData(x, y);//把此方块区域data[][]标志为1,表示有数据
					for (int k = x; k < x + 4; k++) {//循环遍历4层,看是否有哪一层都有方块的情况,以便消除那一行方格和统计得分
						int sum = 0;
						for (int j = 1; j <= 10; j++) {
							if (data[k][j] == 1) {
								sum++;
							}
						}
						if (sum == 10) {//如果k层都有方块,则消除k层方块
							removeRow(k);
						}
					}
					for (int j = 1; j <= 10; j++) {//游戏最上面的4层不能有方块,否则游戏失败
						if (data[3][j] == 1) {
							running = false;
							break;
						}
					}
					break;
				}
				// 如果可以掉落
				x++;// 层加一
				fall(x, y);// 掉下来一层
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
 
		}
	}
	/*判断正下落的方块是否可以下落*/
	public boolean canFall(int m, int n) {
		int temp = 0x8000;//表示1000 0000 0000 0000
		for (int i = 0; i < 4; i++) {//循环遍历16个方格(4*4)
			for (int j = 0; j < 4; j++) {
				if ((temp & rect) != 0) {// 此处有方块时
					if (data[m + 1][n] == 1)// 如果下一个地方有方块,则直接返回false
						return false;
				}
				n++;//列加一
				temp >>= 1;
			}
			m++;// 下一行
			n = n - 4;// 回到首列
		}
		return true;//可以掉落返回true
	}
	/*把不可下降的方块的对应的data存储为1,表示此坐标有方块*/
	public void saveData(int m, int n) {
		int temp = 0x8000;//表示1000 0000 0000 0000
		for (int i = 0; i < 4; i++) {//循环遍历16个方格(4*4)
			for (int j = 0; j < 4; j++) {
				if ((temp & rect) != 0) {// 此处有方块时
					data[m][n] = 1;//data数组存放为1
				}
				n++;//下一列
				temp >>= 1;
			}
			m++;// 下一行
			n = n - 4;// 回到首列
		}
	}
	/*移除row行所有方块,以上的依次往下降*/
	public void removeRow(int row) {
		for (int i = row; i >= 1; i--) {
			for (int j = 1; j <= 10; j++) {
				data[i][j] = data[i - 1][j];//
			}
		}
		reflesh();// 刷新移除row行方块后的游戏主面板区域
		score += 10;// 分数加10;
		label.setText("score: " + score);//显示得分
	}
	/* 刷新移除row行方块后的游戏主面板区域*/
	public void reflesh() {
		for (int i = 1; i < 25; i++) {
			for (int j = 1; j < 11; j++) {
				if (data[i][j] == 1) {//有方块的地方把方块设置为绿色
					grids[i][j].setBackground(Color.GREEN);
				} else {//无方块的地方把方块设置为白色
					grids[i][j].setBackground(Color.WHITE);
				}
			}
		}
	}
	/*方块掉落一层*/
	public void fall(int m, int n) {
		if (m > 0)// 方块下落一层时
			clear(m - 1, n);// 清除上一层有颜色的方块
		draw(m, n);// 重新绘制方块图像
	}
	/*清除方块掉落之前有颜色的地方*/
	public void clear(int m, int n) {
		int temp = 0x8000;//表示1000 0000 0000 0000
		for (int i = 0; i < 4; i++) {//循环遍历16个方格(4*4)
			for (int j = 0; j < 4; j++) {
				if ((temp & rect) != 0) {// 此处有方块时
					grids[m][n].setBackground(Color.WHITE);//清除颜色,变为白色
				}
				n++;//下一列
				temp >>= 1;
			}
			m++;//下一行
			n = n - 4;//回到首列
		}
	}
	/*绘制掉落后方块图像*/
	public void draw(int m, int n) {
		int temp = 0x8000;//表示1000 0000 0000 0000
		for (int i = 0; i < 4; i++) {//循环遍历16个方格(4*4)
			for (int j = 0; j < 4; j++) {
				if ((temp & rect) != 0) {// 此处有方块时
					grids[m][n].setBackground(Color.GREEN);//有方块的地方变为绿色
				}
				n++;//下一列
				temp >>= 1;
			}
			m++;//下一行
			n = n - 4;//回到首列
		}
	}
 
	@Override
	public void keyPressed(KeyEvent e) {
	}
 
	@Override
	public void keyReleased(KeyEvent e) {
	}
 
	@Override
	public void keyTyped(KeyEvent e) {
		if (e.getKeyChar() == 'a') {// 方格进行左移
			if (running == false) {
				return;
			}
			if (y <= 1)//碰到左边墙壁时
				return;
			int temp = 0x8000;//表示1000 0000 0000 0000
			for (int i = x; i < x + 4; i++) {//循环遍历16个方格(4*4)
				for (int j = y; j < y + 4; j++) {
					if ((rect & temp) != 0) {// 此处有方块时
						if (data[i][j - 1] == 1) {//如果左移一格有方块时
							return;
						}
					}
					temp >>= 1;
				}
			}
			clear(x, y);//可以进行左移操作时,清除左移前方块颜色
			y--;
			draw(x, y);//然后重新绘制左移后方块的图像
		}
		if (e.getKeyChar() == 'd') {//方块进行右移操作
			if (running == false) {
				return;
			}
			int temp = 0x8000;
			int m = x, n = y;
			int num = 7;
			for (int i = 0; i < 4; i++) {
				for (int j = 0; j < 4; j++) {
					if ((temp & rect) != 0) {
						if (n > num) {
							num = n;
						}
					}
					temp >>= 1;
					n++;
				}
				m++;
				n = n - 4;
			}
			if (num >= 10) {
				return;
			}
			temp = 0x8000;
			for (int i = x; i < x + 4; i++) {
				for (int j = y; j < y + 4; j++) {
					if ((rect & temp) != 0) {
						if (data[i][j + 1] == 1) {
							return;
						}
					}
					temp >>= 1;
				}
			}
			clear(x, y);//可以进行右移操作时,清除右移前方块颜色
			y++;
			draw(x, y);//然后重新绘制右移后方块的图像
		}
		if (e.getKeyChar() == 's') {//方块进行下移操作
			if (running == false) {
				return;
			}
			if (canFall(x, y) == false) {
				saveData(x, y);
				return;
			}
			clear(x, y);//可以进行下移操作时,清除下移前方块颜色
			x++;
			draw(x, y);//然后重新绘制下移后方块的图像
		}
		if (e.getKeyChar() == 'w') {//改变方块形状
			if (running == false) {
				return;
			}
			int i = 0;
			for (i = 0; i < allRect.length; i++) {//循环遍历19个方块形状
				if (allRect[i] == rect)//找到下落的方块对应的形状,然后进行形状改变
					break;
			}
			if (i == 0)//为正方形方块无需形状改变,为方块图形种类1
				return;
			clear(x, y);
			if (i == 1 || i == 2) {//为方块图形种类2
				rect = allRect[i == 1 ? 2 : 1];
				if (y > 7)
					y = 7;
			}
			if (i >= 3 && i <= 6) {//为方块图形种类3
				rect = allRect[i + 1 > 6 ? 3 : i + 1];
			}
			if (i >= 7 && i <= 10) {//为方块图形种类4
				rect = allRect[i + 1 > 10 ? 7 : i + 1];
			}
			if (i == 11 || i == 12) {//为方块图形种类5
				rect = allRect[i == 11 ? 12 : 11];
			}
			if (i == 13 || i == 14) {//为方块图形种类6
				rect = allRect[i == 13 ? 14 : 13];
			}
			if (i >= 15 && i <= 18) {//为方块图形种类7
				rect = allRect[i + 1 > 18 ? 15 : i + 1];
			}
			draw(x, y);
		}
	}
}

三、游戏测试

以上步骤都做好就可以进行测试了,先激活Fn键,然后ctrl+F11快捷运行。

1.弹出窗口,直接开始游戏,按W键改变方块的方向,按S键快速向下降落,按A键向左移动,按D键向右移动。详细如下图

2.通过最下面一行满格来获得分数。

3.当方块达到顶峰时,则判定游戏结束,弹出下面示例:

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

2024有哪些免费的mac苹果电脑内存清理工具?

在我们日常使用苹果电脑的过程中&#xff0c;随着时间的推移&#xff0c;可能会发现设备的速度变慢了&#xff0c;甚至出现卡顿的现象。其中一个常见的原因就是程序占用内存过多&#xff0c;导致系统无法高效地运行。那么&#xff0c;苹果电脑内存怎么清理呢&#xff1f;本文将…

【机器学习8】采样

1 均匀分布随机数 均匀分布是指整个样本空间中的每一个样本点对应的概率&#xff08;密度&#xff09; 都是相等的。 根据样本空间是否连续&#xff0c; 又分为离散均匀分布和连续均匀分布。编程实现均匀分布随机数生成器一般可采用线性同余法&#xff08;Linear Congruential…

大数据-之LibrA数据库系统告警处理(ALM-12046 网络写包丢包率超过阈值)

告警解释 系统每30秒周期性检测网络写包丢包率&#xff0c;并把实际丢包率和阈值&#xff08;系统默认阈值0.5%&#xff09;进行比较&#xff0c;当检测到网络写包丢包率连续多次&#xff08;默认值为5&#xff09;超过阈值时产生该告警。 用户可通过“系统设置 > 阈值配置…

全功能知识付费变现小程序系统源码 自带流量主 轻松帮你赚钱 带完整搭建教程

大家好啊&#xff0c;今天罗峰要来给大家分享一款全功能知识付费变现小程序源码系统 。近年来互联网技术的快速发展&#xff0c;以及人们对知识付费的需求不断增长。全功能知识付费变现小程序系统源码的出现为大家提供一个全面、高效、安全的解决方案&#xff0c;帮助用户实现知…

阿里云的99元服务器和腾讯云的88元云服务器选择哪个?怎么选?

近日&#xff0c;阿里云宣布在2023年双十一优惠活动中推出了一系列降价措施&#xff0c;使得同配置的云服务器比腾讯云更具竞争力。这一消息不仅在云计算领域引起了轰动&#xff0c;更为广大互联网用户提供了更为实惠的选择。 阿里云推出99元一年的服务器&#xff0c;续费价格…

Linux学习教程(第三章 Linux文件和目录管理)1

第三章 Linux文件和目录管理&#xff08;初识Linux命令&#xff09; 对初学者来说&#xff0c;管理 Linux 系统中的文件和目录&#xff0c;是学习 Linux 至关重要的一步。 为了方便管理文件和目录&#xff0c;Linux 系统将它们组织成一个以根目录 / 开始的倒置的树状结构。Li…

VueEcharts的使用简解以及常用网站

目录 一&#xff1a;前言 二&#xff1a;实现 1、安装echarts依赖 2、创建图表 1&#xff09;全局引入 2&#xff09;按需引入 三&#xff1a;结尾 一&#xff1a;前言 VueEcharts 是项目开发中可视化的一个重要知识部分。其涵盖了柱状图&#xff0c;饼状图&#xff0c;…

机器学习深度学习服务器推荐

大学生、研究生未免找不到还有的GPU服务器&#xff0c;这边博主推荐 https://featurize.cn?s3d13789cb8184f16bb6133b20c353207 方便&#xff0c;便宜&#xff0c;不会自动删除上传项目文件&#xff0c;支持VScode、pycharm、SSH链接&#xff0c;上传文件速度快。强烈推荐&am…

B031-网络编程 Socket Http TomCat

目录 计算机网络网络编程相关术语IP地址ip的概念InerAdress的了解与测试 端口URLTCP、UDP和7层架构TCPUDPTCP与UDP的区别和联系TCP的3次握手七层架构 Socket编程服务端代码客户端代码 http协议概念Http报文 Tomcat模拟 计算机网络 见文档 网络编程相关术语 见文档 IP地址 …

【Proteus仿真】【STM32单片机】锂电池管理系统

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真STM32单片机控制器&#xff0c;使用LCD1602显示模块、DS18B20温度传感器、PCF8691 ADC模块、按键、LED蜂鸣器模块等。 主要功能&#xff1a; 系统运行后&#xff0c;LCD1602显示温…

2核2G3M带宽云服务器99元(续费同价),阿里云老用户可买!

在阿里云的双11云服务器活动中&#xff0c;用户对轻量服务器2核2G3M带宽和经济型e实例2核2G配置3M带宽特别关注。除了这两款产品&#xff0c;阿里云还提供了其他性价比很高的云服务器配置&#xff0c;让用户有更多的选择。 经济型e实例2核2G3M配置99元一年是适用于个人和普通企…

《向量数据库指南》——TruLens 用于语言模型应用跟踪和评估

TruLens 用于语言模型应用跟踪和评估 TruLens 是一个用于评估语言模型应用(如 RAG)的性能的开源库。通过 TruLens,我们还可以利用语言模型本身来评估输出、检索质量等。 构建语言模型应用时,多数人最关心的问题是 AI 幻觉(hallucination)。RAG 通过为语言模型提供检索上下文…

ython requests 库中 POST 方法的使用及文档改进

在使用Python库requests的过程中&#xff0c;发起者r1chardj0n3s遇到了一个关于POST方法的问题。他表示&#xff0c;他花费了一些时间才弄清楚requests.post函数的使用方法。然而&#xff0c;在requests的官方文档中&#xff0c;关于POST方法的描述只出现在API文档中&#xff0…

基于springboot实现医患档案管理系统项目【项目源码】计算机毕业设计

基于springboot实现医患档案管理系统演示 Java语言简介 Java是由SUN公司推出&#xff0c;该公司于2010年被oracle公司收购。Java本是印度尼西亚的一个叫做爪洼岛的英文名称&#xff0c;也因此得来java是一杯正冒着热气咖啡的标识。Java语言在移动互联网的大背景下具备了显著的…

【博客系统】 一

该博客系统基于servlet和mysql数据库 , 并且通过xshell终端工具部署至云服务器. 实现的功能包括: 1.博客列表页 2.博客详情页 3.登陆页面 4.强制登陆检查 5.获取用户信息 6.退出登陆 7.发布博客 一.系统展示 登陆页面 博客列表页 博客详情页 博客编辑页 下面就开始编写代码了.…

【Linux】Linux进程间通信(一)

​ ​&#x1f4dd;个人主页&#xff1a;Sherry的成长之路 &#x1f3e0;学习社区&#xff1a;Sherry的成长之路&#xff08;个人社区&#xff09; &#x1f4d6;专栏链接&#xff1a;Linux &#x1f3af;长路漫漫浩浩&#xff0c;万事皆有期待 上一篇博客&#xff1a;【Linux】…

如何选择适合企业的数字化看板系统

数字化看板是一种数字化管理工具&#xff0c;它具有实时数据展示、任务分配与优先级管理、问题追踪与解决、供应链管理与协同以及数据分析与预测等功能。这些功能可以帮助企业提高生产效率、降低成本、优化资源配置并实现生产过程的透明化。 具体来说&#xff0c;数字化看板可以…

Navicat 使用

安装包请私信本人 软件安装 需要注意以下不要先运行navicat软件,先运行 在工具的第一个选项(1.Patch)里选择Backup&#xff0c;然后点击&#xff0c;Patch按钮&#xff0c;选择Navicat的安装位置中的navicat.exe文件 如果显示这样表示成功了 在这选择语言简体中文 点击Gener…

第五章 路由技术及应用

目录 5.1 直连路由概述 5.1.1 直连路由工作原理 5.1.2 直连路由配置 5.2 直连路由仿真 5.3 静态路由技术 5.3.1 静态路由定义 5.3.2 静态路由工作原理 5.3.3 静态路由配置 5.3.4 默认路由 (1) 默认路由概述 (2) 配置默认路由 (3) 默认路由应用场合&#xff1a;上网…

基于springboot实现疫苗接种管理系统项目【项目源码】计算机毕业设计

基于springboot实现疫苗接种管理系统演示 Java语言简介 Java是由SUN公司推出&#xff0c;该公司于2010年被oracle公司收购。Java本是印度尼西亚的一个叫做爪洼岛的英文名称&#xff0c;也因此得来java是一杯正冒着热气咖啡的标识。Java语言在移动互联网的大背景下具备了显著的…