Java ——简易俄罗斯方块

article/2025/11/2 7:18:21

一、将对象抽象为类

首先考虑俄罗斯方块游戏中含有哪些具体的对象,对象中含有哪些具体属性和方法,然后用代码来实现。

建立如下类:

Cell类:代表最小的方格单位,构成7种图形的最基本图形。

    含有row(行号),col(列号),image(对应的图片)属性,

    含有left(左移),right(右移),drop(下落)方法。

Tetromino类:代表由4个最小方格构成的7种图形的合集。

    含有cells(四个方块)属性,

    含有moveLeft(四格方块向左移动),moveRight(四格方块向右移动),softDrop(软下落),randomOne(随机生成一个四格方格)方法。

T类继承于Tetromino类:

I类继承于Tetromino类:

L类继承于Tetromino类:

S类继承于Tetromino类:

Z类继承于Tetromino类:

O类继承于Tetromino类:

J类继承于Tetromino类:

Tetris类:俄罗斯方块的主方法类,包括了游戏运行过程中所需要的众多方法。

    含有currentOne(正在下落的四格方块),nextOne(即将下落的四格方块),Cell[][]wall(二维数组的表格,代表墙)属性。

二、类的实现

Notes:各类实现过程中要符合Javabean规范。

Cell类:

package com.tetris;import java.awt.image.BufferedImage;/** 俄罗斯方块中的最小单位:方格* 特征(属性):* row--行号* col--列号* image--对应的图片* * 行为(方法)* 	left();* 	right();* 	drop();*/
public class Cell {private int row;	//行private int col;	//列private BufferedImage image;public Cell(int row, int col, BufferedImage image) {super();this.row = row;this.col = col;this.image = image;}public Cell() {super();// TODO Auto-generated constructor stub}public int getRow() {return row;}public void setRow(int row) {this.row = row;}public int getCol() {return col;}public void setCol(int col) {this.col = col;}public BufferedImage getImage() {return image;}public void setImage(BufferedImage image) {this.image = image;}@Overridepublic String toString() {return "(" + row + ", " + col + ")";}//向左移动public void left(){col--;}//向右移动public void right(){col++;}//向下移动public void drop(){row++;}
}
Tetromino类:
package com.tetris;import java.util.Arrays;import javax.xml.transform.Templates;/** 四格方块* 属性:* ---cells,----四个方块* * 行为:* 	moveLeft()* 	moveRight()* 	softDrop()*/
public class Tetromino {protected Cell[] cells=new Cell[4];//四格方块向左移动//实际上:就是每个方块向左移动public void moveLeft(){for (int i = 0; i < cells.length; i++) {cells[i].left();}}//四格方块向右移动//实际上:就是每个方块向右移动public void moveRight(){for (int i = 0; i < cells.length; i++) {cells[i].right();}}//四格方块向下移动//实际上:就是每个方块向下移动public void softDrop(){for (int i = 0; i < cells.length; i++) {cells[i].drop();}}@Overridepublic String toString() {return "[" + Arrays.toString(cells) + "]";}//随机生成一个四格方块public static Tetromino randomOne(){Tetromino t = null;int num=(int)(Math.random()*7);switch (num){case 0:t=new T();break;case 1:t=new O();break;case 2:t=new I();break;case 3:t=new J();break;case 4:t=new L();break;case 5:t=new S();break;case 6:t=new Z();break;default:break;}return t;}
}

T类继承于Tetromino类:

package com.tetris;public class T extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public T(){cells[0]=new Cell(0,4,Tetris.T);cells[1]=new Cell(0,3,Tetris.T);cells[2]=new Cell(0,5,Tetris.T);cells[3]=new Cell(1,4,Tetris.T);}
}

I类继承于Tetromino类:

package com.tetris;public class I extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public I(){cells[0]=new Cell(0,4,Tetris.I);cells[1]=new Cell(0,3,Tetris.I);cells[2]=new Cell(0,5,Tetris.I);cells[3]=new Cell(0,6,Tetris.I);}
}

L类继承于Tetromino类:

package com.tetris;public class L extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public L(){cells[0]=new Cell(0,4,Tetris.L);cells[1]=new Cell(0,3,Tetris.L);cells[2]=new Cell(0,5,Tetris.L);cells[3]=new Cell(1,5,Tetris.L);}
}

S类继承于Tetromino类:

package com.tetris;public class S extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public S(){cells[0]=new Cell(1,4,Tetris.S);cells[1]=new Cell(0,3,Tetris.S);cells[2]=new Cell(0,4,Tetris.S);cells[3]=new Cell(1,5,Tetris.S);}
}

Z类继承于Tetromino类:

package com.tetris;public class Z extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public Z(){cells[0]=new Cell(0,4,Tetris.Z);cells[1]=new Cell(0,5,Tetris.Z);cells[2]=new Cell(1,3,Tetris.Z);cells[3]=new Cell(1,4,Tetris.Z);}
}

O类继承于Tetromino类:

package com.tetris;public class O extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public O(){cells[0]=new Cell(0,4,Tetris.O);cells[1]=new Cell(0,5,Tetris.O);cells[2]=new Cell(1,4,Tetris.O);cells[3]=new Cell(1,5,Tetris.O);}
}

J类继承于Tetromino类:

package com.tetris;public class J extends Tetromino {//提供构造器,进行初始化//T型的四格方块的位置public J(){cells[0]=new Cell(0,4,Tetris.J);cells[1]=new Cell(0,3,Tetris.J);cells[2]=new Cell(0,5,Tetris.J);cells[3]=new Cell(1,3,Tetris.J);}
}

Tetris类:

	 //属性:正在下落的四格方块private Tetromino currentOne=Tetromino.randomOne();//属性:将要下落的四格方块private Tetromino nextOne=Tetromino.randomOne();		 //属性:墙,20行10列的表格  宽度为26private Cell[][]wall=new Cell[20][10];

三、绘制俄罗斯方块图形

    个人理解,这个过程就是显现出游戏界面的过程,当然啦,这一步主要是加载静态资源,诸如图片,音频和视频等。

1.加载静态资源

俄罗斯方块主要应用的静态资源是图片,所以我们用到的是IO类中的ImageIO类中的ImageIO.read方法,导入各类四格方块的图形图片以及背景图片,具体代码如下:

public static  BufferedImage T;public static  BufferedImage I;public static  BufferedImage O;public static  BufferedImage J;public static  BufferedImage L;public static  BufferedImage S;public static  BufferedImage Z;public static  BufferedImage background;static{try {/** getResouce(String url)* url:加载图片的路径* 相对位置是同包下*/T=ImageIO.read(Tetris.class.getResource("T.png"));I=ImageIO.read(Tetris.class.getResource("I.png"));O=ImageIO.read(Tetris.class.getResource("O.png"));J=ImageIO.read(Tetris.class.getResource("J.png"));L=ImageIO.read(Tetris.class.getResource("L.png"));S=ImageIO.read(Tetris.class.getResource("S.png"));Z=ImageIO.read(Tetris.class.getResource("Z.png"));background=ImageIO.read(Tetris.class.getResource("tetris.png"));} catch (Exception e) {e.printStackTrace();}}

2.画游戏静态界面

在这一部分中需要绘制三部分,用到了三种方法,分别是paintCurrentOne(正在下落的四格方块),paintNextOne(等待进入的四格方块),paintWall(背景墙)。

绘制需要重写JPanel类中的paint(Graphics g)方法,具体代码实现如下:

public void paint(Graphics g){//绘制背景/** g:画笔* g.drawImage(image,x,y,null)* x:开始绘制的横坐标* y:开始绘制的纵坐标*/g.drawImage(background,0,0,null);//平移坐标轴g.translate(15, 15);//绘制墙paintWall(g);//绘制正在下落的四格方块paintCurrentOne(g);//绘制下一个即将下落的四格方块paintNextOne(g);}/** 绘制下一个即将下落的四格方块* 绘制到面板的右上角的相应区域*/public void paintNextOne(Graphics g){//获取nextOne对象的四个元素Cell[] cells=nextOne.cells;for (Cell c:cells) {//获取每一个元素的行号和列号int row=c.getRow();int col=c.getCol();//横坐标和纵坐标int x=col*CELL_SIZE+260;int y=row*CELL_SIZE+26;g.drawImage(c.getImage(), x, y, null);}}/** 绘制正在下落的四格方块* 取出数组的元素* 绘制数组的图片* 横坐标x* 纵坐标y*/public void paintCurrentOne(Graphics g){Cell[] cells=currentOne.cells;for (Cell c:cells) {int x=c.getCol()*CELL_SIZE;int y=c.getRow()*CELL_SIZE;g.drawImage(c.getImage(), x, y, null);}}/** 墙是20行,10列的表格* 是一个二维数组* 用双层循环* 绘制正方形*/public void paintWall(Graphics a){//外层循环控制行数for (int i = 0; i < 20; i++) {//内层循环控制列数for (int j = 0; j < 10; j++) {int x=j*CELL_SIZE;int y=i*CELL_SIZE;Cell cell=wall[i][j];a.drawRect(x, y, CELL_SIZE, CELL_SIZE);if(wall[i][j]==null){a.drawRect(x, y, CELL_SIZE, CELL_SIZE);}else{a.drawImage(cell.getImage(),x,y,null);}}}}

实现效果如下:

3.让四格方块动起来

光有静态的画面是不能够称为游戏的,还有要动态效果和接收键盘指令并响应的能力。

(1)动态效果

俄罗斯方块中的动态效果主要指7种四格方块拥有自动下降,软下降,左移,右移,旋转的能力,分别使用canDrop(),softDropAction(),moveLeftAction(),moveRightAction(),spinCellAction()方法来实现,与此同时,还需根据游戏规则注意四格方块可能遇到触碰到左右边界,方块覆盖等错误,在此使用outOfBounds(),coincide()方法来避免。当不能下落时,需要将四格方块,嵌入到墙中,使用landToWall()方法。

具体代码实现如下:

	/** 使用left键控制向左的行为*/public void moveLeftAction() {currentOne.moveLeft();if(outOfBounds()||coincide()){currentOne.moveRight();}}/** 使用right键控制向右的行为*/public void moveRightAction() {currentOne.moveRight();if(outOfBounds()||coincide()){currentOne.moveLeft();}}/** 使用down键控制四格方块的下落*/public void softDropAction() {if(canDrop()){currentOne.softDrop();}else{landToWall();currentOne=nextOne;nextOne=Tetromino.randomOne();}}public boolean outOfBounds(){Cell[] cells=currentOne.cells;for (Cell c : cells) {int col=c.getCol();if(col<0||col>9){return true;}}return false;	}public boolean coincide(){Cell[] cells=currentOne.cells;for (Cell c : cells) {int row=c.getRow();int col=c.getCol();if(wall[row][col]!=null){return true;}}return false;}public boolean canDrop(){Cell[] cells=currentOne.cells;for (Cell c: cells) {//获取每个元素的行号/** 判断:* 只要有一个元素的下一行上有方块* 或者只要有一个元素到达最后一行,就不能下落了*/int row=c.getRow();int col=c.getCol();if(row==19){return false;}if(wall[row+1][col]!=null){return false;}}return true;}/** 当不能下落时,需要将四格方块,嵌入到墙中* 也就是存储到二维数组中相应的位置上*/public void landToWall(){Cell[] cells=currentOne.cells;for (Cell c : cells) {//获取最终的行号和列号int row=c.getRow();int col=c.getCol();wall[row][col]=c;}}

实现效果如下:


(2)接收键盘指令并响应

游戏和玩家紧密关联,所以接下来我们需要使玩家能够通过键盘控制四格方块移动。

因此,我们要开启键盘监听来达到玩家实时控制游戏的目的,并且通过不同的按键调用四格方块移动的不同方法。

具体代码如下:

//开启键盘监听事件KeyListener l=new KeyAdapter() {public void keyPressed(KeyEvent e){//获取以下键子的代号int code=e.getKeyCode();switch (code) {case KeyEvent.VK_DOWN:softDropAction();break;case KeyEvent.VK_LEFT:moveLeftAction();break;case KeyEvent.VK_RIGHT:moveRightAction();break;	}repaint();}};this.addKeyListener(l);this.requestFocus();while(true){/** 当程序运行到此,会进入睡眠状态* 睡眠时间为300毫秒,单位为毫秒* 300毫秒后会自动执行后续代码*/try {Thread.sleep(300);} catch (InterruptedException e) {e.printStackTrace();}if(canDrop()){currentOne.softDrop();}else{landToWall();//将下一个下落的四格方块赋值给正在下落的变量currentOne=nextOne;nextOne=Tetromino.randomOne();}/** 下落之后,要重新进行绘制,才会看到下落后的位置* repaint方法也是Jpanel类中提供的* 此方法调用了paint方法*/repaint();}}
实现效果如下:




http://chatgpt.dhexx.cn/article/cIFSqRkn.shtml

相关文章

【Java】俄罗斯方块小游戏(附源码)

俄罗斯方块小游戏 一、最终效果二、功能需求二、程序实现 一、最终效果 二、功能需求 1、 在二维平面里面用各种随机产生的方块堆积木&#xff0c;每满一行消去一行&#xff0c;当达到顶部时&#xff0c;游戏结束。 2、 通过方向键来控制方块转动&#xff0c;左移&#xff0c;…

Java实现俄罗斯方块附源码

Java实现俄罗斯方块 软件的开发过程&#xff1a; 1&#xff09; 搞清需求&#xff0c;就是软件的功能描述 俄罗斯方块的基本规则&#xff1a; 1、一个用于摆放小型正方形的平面虚拟场地&#xff0c;其标准大小&#xff1a; 行宽为10&#xff0c;列高为20&#xff0c;以每个小…

JAVA 实现《俄罗斯方块》游戏|CSDN创作打卡

前言 俄罗斯方块是一个最初由阿列克谢帕吉特诺夫在苏联设计和编程的益智类视频游戏。 《俄罗斯方块》的基本规则是移动、旋转和摆放游戏自动输出的各种方块&#xff0c;使之排列成完整的一行或多行并且消除得分。 用java语言实现&#xff0c;采用了swing技术进行了界面化处理…

Java游戏开发——俄罗斯方块

游戏介绍 俄罗斯方块是一款风靡全球的电视游戏机和掌上游戏机的游戏&#xff0c;它曾经造成的轰动与造成的经济价值可以说是游戏史上的一件大事。这款游戏最初是由苏联的游戏制作人Alex Pajitnov制作的&#xff0c;它看似简单但却变化无穷&#xff0c;游戏过程中仅需要玩家将不…

Java写俄罗斯方块,了解一下

Java俄罗斯方块目录&#xff1a; Java俄罗斯方块 ---&#xff08;一&#xff09;游戏场景篇Java俄罗斯方块 ---&#xff08;二&#xff09;游戏操作与逻辑篇Java写俄罗斯方块&#xff08;完整版&#xff09; 简要分析&#xff1a; 俄罗斯方块的规则在这里就不细说了&#xf…

Java游戏开发 —— 俄罗斯方块

引言&#xff1a; 俄罗斯方块的代码实现很简单&#xff0c;很有意思&#xff01; 思路&#xff1a; 1、创建主窗口&#xff0c;加载菜单及游戏面板。 2、在游戏面板中初始化各种参数&#xff0c;并建立各种功能组件。 3、利用paint()函数开始画方块。 4、游戏结束&#xff0c;…

Java俄罗斯方块,老程序员花了一个周末,连接中学年代!

Java俄罗斯方块&#xff0c;老程序员花了一个周末&#xff0c;连接中学年代&#xff01; 热门专栏推荐 【1】Java小游戏&#xff08;俄罗斯方块、飞机大战、植物大战僵尸等&#xff09; 【2】JavaWeb项目实战&#xff08;图书管理、在线考试、宿舍管理等&#xff09; 【3】Jav…

软件设计实战:基于Java的俄罗斯方块游戏【完整版】

个人简介 &#x1f468;&#x1f3fb;‍&#x1f4bb;个人主页&#xff1a;陈橘又青 &#x1f3c3;&#x1f3fb;‍♂️博客记录心情&#xff0c;代码编写人生。 &#x1f31f;如果文章对你有用&#xff0c;麻烦关注点赞收藏走一波&#xff0c;感谢支持&#xff01; &#x1f3…

Java实现游戏开发《俄罗斯方块》

一、用Java实现俄罗斯方块游戏&#xff1a; 1、效果图&#xff0c;如下图所示&#xff1a; 7种形态的第一种形态, 如下所示&#xff1a;分布是 &#xff1a;|、S、Z、J、O、L、T; 0 1 0 0 0 1 1 0 1 1 0 0 0 1 0 0 1 1 0 0 1 0 0 0 1 1 1 0 0 1 0 0 1 1 0 0 0 1 1 0 0 1 0 0 1…

俄罗斯方块游戏的设计与实现(Java+Swing+Eclipse)

目录 基于Java的俄罗斯方块游戏的设计与实现 I 摘 要 I Based on the design and implementation of Java game Tetris II Abstract II 1 绪论 1 1.1程序开发背景及意义 1 1.2开发技术概述 2 1.3俄罗斯方块游戏的研究现状 2 1.3.1 国内外研究现状 2 1.3.2 文献综述 3 2相关技术…

【Java小游戏】俄罗斯方块

文章目录 规则准备工作编写小方块类编写四方格父类创建7种不同的形状编写俄罗斯方块主类初始化7种形状 随机生成四方格创建游戏场景绘制游戏绘制游戏背景绘制游戏主区域绘制正在下落的四方格绘制下一个下落的四方格绘制游戏得分绘制游戏状态 编写游戏逻辑判断方块是否出界判断方…

Java实现俄罗斯方块小游戏。(附完整源代码)

大家好&#xff0c;我是百思不得小赵。 创作时间&#xff1a;2022 年 5 月 12 日 博客主页&#xff1a; &#x1f50d;点此进入博客主页 —— 新时代的农民工 &#x1f64a; —— 换一种思维逻辑去看待这个世界 &#x1f440; 今天是加入CSDN的第1167天。觉得有帮助麻烦&#x…

Java实现俄罗斯方块游戏(简单版)

游戏页面效果如下&#xff1a; 俄罗斯方块游戏本身的逻辑&#xff1a; 俄罗斯方块游戏的逻辑是比较简单的。它就类似于堆砌房子一样&#xff0c;各种各样的方地形状是不同的。但是&#xff0c;俄罗斯方块游戏的界面被等均的分为若干行和若干列&#xff0c;因此方块的本质就是占…

vs2019功能介绍_MFC界面库BCGControlBar v30.0新功能详解:支持VS 2019

亲爱的BCGSoft用户&#xff0c;我们非常高兴地宣布BCGControlBar Professional for MFC和BCGSuite for MFC v30.0正式发布&#xff01;新版本添加了对Visual Studio 2019的支持等。接下来几篇文章将对这个版本的新功能一一进行介绍&#xff0c;让您对BCG这个控件有一个全新的认…

MFC界面库BCGControlBar v32.0 - 网格、报表控件升级

亲爱的BCGSoft用户&#xff0c;我们非常高兴地宣布BCGControlBar Professional for MFC和BCGSuite for MFC v32.0正式发布&#xff01;新版本支持Windows 11、增强功能区简化模式、改进控件外观等&#xff0c;以及其他新功能和改进。需要最新版的可以点击这里【BCG下载】 BCGC…

MFC扩展库BCGControlBar :网格和报告控件

BCGControlBar ("Business Components Gallery ControlBar")是MFC扩展库&#xff0c;使您可以创建具有完全自定义选项&#xff08;功能区、可自定义工具栏、菜单等&#xff09;以及一组专业设计的丰富Microsoft Office和Microsoft Visual Studio的应用程序 GUI控件&a…

MFC界面库之BCGControlers使用

觉得默认的MFC实在是太丑了,想要用一下扩展的界面库.网上查了些资料.不过在这里我还是小结一下使用的过程吧! 1.当然是安装文件了,在上一步不用多勾选,直接按照默认的就可以了...但是要注意一定要以管理员身份运行BCGCBProBuildWizard.exe否则没有办法通过...... 2.配置lib文件…

MFC界面控件BCGControlBar v33.3 - 编辑控件功能升级

BCGControlBar库拥有500多个经过全面设计、测试和充分记录的MFC扩展类。 我们的组件可以轻松地集成到您的应用程序中&#xff0c;并为您节省数百个开发和调试时间。 BCGControlBar专业版和BCGSuite for MFC v33.3已正式发布了&#xff0c;该版本包含了增强的Ribbon自定义、新的…

MFC界面美化

排列整齐 基于MFC编写GUI代码时&#xff0c;界面美化最基本的部分就是排列整齐&#xff0c;如果是用Visual Studio 2015 IDE 开发&#xff0c;那就十分方便了&#xff0c;在快捷功能框即有相关按钮&#xff0c;这和Qt的控件调整有些类似&#xff0c;可以有效减少我们在布局上耗…

MFC界面库BCGControlBar v30.0新功能详解:支持VS 2019

亲爱的BCGSoft用户&#xff0c;我们非常高兴地宣布BCGControlBar Professional for MFC和BCGSuite for MFC v30.0正式发布&#xff01;新版本添加了对Visual Studio 2019的支持等。接下来几篇文章将对这个版本的新功能一一进行介绍&#xff0c;让您对BCG这个控件有一个全新的认…