java“俄罗斯方块”

首先新建议一个包为Tetris (俄罗斯方块)

类名也叫做Tetris;

代码运行:

package Tetris;
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 Tetris 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 Tetris() {
		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) {
		Tetris m = new Tetris(); //创建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键向右移动。

 

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

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

相关文章

Rust图形界面:eGUI的Panel布局

文章目录 Panel布局尺寸调节源码 Panel布局 eGUI提供面板堆叠的布局方案&#xff0c;即Panel布局。其布局逻辑是&#xff0c;根据当前面板指定的方向&#xff0c;尽可能地填充空间。 CentralPanel 占据屏幕剩余部分的空间SidePanel 占据屏幕两侧的空间&#xff0c;在具体调用…

听GPT 讲Rust源代码--library/core/src(5)

题目来自 Understanding Box in Rust &#x1f980; File: rust/library/core/src/num/saturating.rs 在Rust的核心库中&#xff0c;源代码路径rust/library/core/src/num/saturating.rs所对应的文件是用来实现饱和运算的功能。 饱和运算是一种数值运算的方式&#xff0c;用于处…

中级程序员——uniapp和小程序面试题

&#x1f604;博主&#xff1a;小猫娃来啦 &#x1f604;文章核心&#xff1a;uniapp和小程序面试题 文章目录 用uniapp有遇到一些兼容性问题吗&#xff1f;uniapp最大的优点是什么&#xff1f;uniapp如何实现多端兼容&#xff1f;uniapp是如何做跨端适配的&#xff1f;常用的u…

1~2亿条数据需要缓存之安装redis集群(哈希取余分区、一致性哈希算法分区、哈希槽分区)

安装redis集群 面试题 1~2亿条数据需要缓存&#xff0c;请问如何设计这个存储案例??? 回答: 单机单台100%不可能&#xff0c;肯定是分布式存储&#xff0c;用redis如何落地&#xff1f; 上述问题阿里P6~P7工程案例和场景设计类必考题目&#xff0c; 一般业界有3种解决方案 …

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

第三章 Linux文件和目录管理(初识Linux命令) 十一、Linux 删除空目录(rmdir命令) Linux rmdir命令:删除空目录 和 mkdir 命令(创建空目录)恰好相反,rmdir(remove empty directories 的缩写)命令用于删除空目录,此命令的基本格式为: [root@localhost ~]# rmdir […

走进ERP管理系统:企业信息化的加速器

走进ERP管理系统&#xff1a;企业信息化的加速器 企业信息化已成为企业提高效率、降低成本、优化资源配置的重要手段。ERP管理系统作为企业信息化的重要组成部分&#xff0c;已经成为了越来越多企业的必备工具。本文将带您走进ERP管理系统&#xff0c;探讨其优势、应用场景以及…

Redis篇---第三篇

系列文章目录 文章目录 系列文章目录前言一、为什么 Redis 需要把所有数据放到内存中?二、Redis 的同步机制了解是什么?三、pipeline 有什么好处,为什么要用 pipeline?前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到…

使用html2canvas插件进行页面截屏

使用纯html实现过程 <!DOCTYPE html> <html><head><title>使用html2canvas生成网页截图</title><script src"https://html2canvas.hertzen.com/dist/html2canvas.min.js"></script> </head><body><h1>…

从Prefetch到Stream:重构v1.0代码库中的流式请求问题与解决方案

问题背景 在进行v1.0代码库的重构时&#xff0c;我们发现当前的prefetch参数存在一些问题。因此&#xff0c;我们计划将prefetch参数替换为stream。同时&#xff0c;我们决定在所有上传的操作中使用流式传输。 然而&#xff0c;是否需要对所有上传操作都进行流式传输存在一些疑…

设置chunk自动扩展到多大

1. 设置chunk自动扩展 execute function task(modify chunk extendable on,8); 2. 设置dbs扩展到多大合适 execute function task(modify space sp sizes,testdb1024,1024,10240) testdb 初始1MB 下次扩1MB 最大10MB

「Java开发指南」如何在Spring中使用JAX-WS注释器?

本文将指导您如何使用JAX-WS注释器从Spring服务生成JAX-WS Web服务&#xff0c;在本教程中&#xff0c;您将学习如何&#xff1a; 为Spring服务启用JAX-WS部署应用程序并测试服务 所有与Spring scaffolding相关的任务都需要MyEclipse Spring或Bling授权。 MyEclipse v2023.1…

成都瀚网科技有限公司抖音带货是否靠谱

成都瀚网科技有限公司&#xff0c;作为一家专业的科技公司&#xff0c;近年来积极参与了抖音带货的浪潮&#xff0c;凭借其出色的产品和服务&#xff0c;赢得了广大用户的信赖。 在当今的数字化时代&#xff0c;抖音带货已经成为了一种非常流行的购物方式。消费者可以通过抖音平…

elementui 实现树形控件单选

实现&#xff1a; <!--author: itmacydesc: 树节点单选 --> <template><div class"about"><el-tree :data"data"ref"tree":props"defaultProps"node-key"id"show-checkboxcheck-strictlycheck-change…

解决SSH连接自动断开的问题

一、环境 centos7.6 xshell7 二、目标 ssh长联状态&#xff0c;不主动断开 三、实施 1.修改/etc/profile文件 末尾添加export TMOUT0 vim /etc/profileexport TMOUT02.修改/etc/ssh/sshd_config文件 默认都被注释掉&#xff0c;放开并设置CAI参数为0-60间 vim /etc…

Python中表格插件Tabulate的用法

目录 一、引言 二、Tabulate插件安装与导入 三、Tabulate基本用法 1、创建表格&#xff1a; 2. 格式化表格&#xff1a; 3. 表格转置&#xff1a; 4、合并单元格&#xff1a; 5、指定每列的格式&#xff1a; 6、指定每行的格式&#xff1a; 7、使用自定义表格格式&am…

kubernetes学习笔记-概念

参考&#xff1a;https://kubernetes.io/zh-cn/docs/concepts/overview/ 概述 Kubernetes 是一个可移植、可扩展的开源平台&#xff0c;用于管理容器化的工作负载和服务&#xff0c;可促进声明式配置和自动化。 Kubernetes 拥有一个庞大且快速增长的生态&#xff0c;其服务、…

观测云产品更新 | 监控、图表、服务管理、单点登录、Pipeline 等优化

观测云更新 监控相关的若干使用优化 1.【告警沉默】功能回归&#xff0c;您仍可以通过告警沉默配置定义重复告警通知的时间间隔。 2.【监控器】删除 ”紧急“、”重要“、”警告“触发条件必须配置任意一项的限制&#xff0c;您可任意配置”紧急“、”重要“、”警告“、”无…

实时云渲染与带宽探讨:解密数字新基建中的网络技术

随着“数字新基建”和5G技术的蓬勃发展&#xff0c;数字孪生、智慧工厂、智慧城市、虚拟仿真教学等领域正经历着迅速的技术进步。部署方案也由最初的本地部署&#xff0c;发展到webGL本地网络方式再到最新的实时云渲染技术。本文将深入探讨实时云渲染技术方案对于带宽的要求条件…

【用unity实现100个游戏之15】开发一个类保卫萝卜的Unity2D塔防游戏4(附项目源码)

文章目录 先看本次实现的最终效果前言把敌人和炮塔全部配置成预制体炮塔商店打开商店放置炮塔升级炮塔出售显示炮塔攻击范围显示玩家金额和血量关闭升级面板和商店功能源码完结 先看本次实现的最终效果 前言 本期紧接着上一篇&#xff0c;本期主要内容是实现商店、购买、出售、…

Python学习笔记--自定义元类

四、自定义元类 到现在&#xff0c;我们已经知道元类是什么鬼东西了。 那么&#xff0c;从始至终我们还不知道元类到底有啥用。 只是了解了一下元类。 在了解它有啥用的时候&#xff0c;我们先来了解下怎么自定义元类。 因为只有了解了怎么自定义才能更好的理解它的作用。…