Java实现俄罗斯方块的源码分享

本文实现的功能有:

1、 初始化游戏窗口

2、初始化游戏的界面

3、初始化游戏的说明面板

4、随机生成下落方块

5、方块下落速度变化

6、判断方块是否可以下落

7、移除某一行方块上面的方块后让上面的方块掉落

8、刷新移除某一行方块后的界面

9、清除方块

10、绘制方块

11、键盘控制方块的移动、变形和快速下落

12、游戏的暂停功能

三小时纯手工打造,具体实现代码:

import javax.swing.*;
import java.awt.*;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.util.Random;

public class start extends JFrame implements KeyListener {
    public static void main(String[] args) throws Exception {
     //   SwingUtilities.invokeLater(start::initWindow);
        start t=new start();
        t.game_begin();
    }
    //游戏的行数为26,列数为12
    private static final int game_x=26;
    private static final int game_y=12;

    //文本域数组
    JTextArea[][] text;
    //二维数组
    int [][] data;
    //显示游戏状态的标签
    JLabel Label1;
    //显示游戏分数的标签
    JLabel Label;
    //提示暂停键的标签
    JLabel label;
    //用于判断游戏是否结束
    boolean isrunning;
    //用于存储所有方块的数组
    int [] allRect;
    //用于存储当前方块的变量
    int rect;
    //线程的休眠时间
    int time=1000;
    //表示方块坐标
    int x,y;
    //该变量用于计算得分
    int score=0;
    //定义一个标志变量,判断游戏是否暂停
    boolean game_pause=false;
    //定义一个变量,用于记录按下暂停的次数
    int pause=0;

    public void initWindow(){
        //设置窗口大小
        this.setSize(600,850);
        //设置窗口是否可见
        this.setVisible(true);
        //设置窗口居中
        this.setLocationRelativeTo(null);
        //设置释放窗体
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        //设置窗体大小不可变
        this.setResizable(false);
        //设置标题
        this.setTitle("俄罗斯方块");

    }
    public start(){
        text=new JTextArea[game_x][game_y];
        data=new int[game_x][game_y];
        //初始化游戏状态的标签
        Label1 =new JLabel("游戏状态:正在游戏中!");
        //初始化游戏分数的标签
        Label=new JLabel("游戏得分:0");
        //初始化提示标签
        label=new JLabel("按下s键,即可暂停游戏!");
        initGamePanel();
        initExplainPanel();
        initWindow();
        //初始化游戏开始的标志
        isrunning=true;
        //初始化存放方块的数组
        allRect =new int[]{0x00cc,0x8888,0x000f,0x888f,0xf888,0xf111,0x111f,0x0eee,0xffff,0x0008
                ,0x0888,0x000e,0x0088,0x000c,0x08c8,0x00e4,0x04c4,0x004e,0x08c4,0x006c,0x04c8,0x00c6};
    }
    //初始化游戏界面
    public void initGamePanel(){
        JPanel game_main=new JPanel();
        game_main.setLayout(new GridLayout(game_x,game_y,1,1));

        for (int i = 0; i < text.length; i++) {
            for (int j = 0; j < text[i].length; j++) {
                //设置文本域的行列数
                text[i][j]=new JTextArea(game_x,game_y);
                //设置文本域的背景颜色
                text[i][j].setBackground(Color.WHITE);//白色
                //添加键盘监听事件
                text[i][j].addKeyListener(this);
                //初始化游戏边界
                if(j==0 || j==text[i].length-1 || i==text.length-1){
                    text[i][j].setBackground(Color.BLACK);//设置为黑色,这里看个人喜好设置
                    data[i][j]=1;//表示这里有方块
                }
                //设置文本域不可编辑
                text[i][j].setEditable(false);
                //文本区域添加到主面板上去
                game_main.add(text[i][j]);
            }
        }
        //将主面板添加到窗口中
        this.setLayout(new BorderLayout());
        this.add(game_main,BorderLayout.CENTER);//把游戏区域添加到窗口的中间
    }
    //初始化游戏的说明界面
    public void initExplainPanel(){
        //创建游戏的左说明面板
        JPanel explain_left=new JPanel();
        //创建游戏的右说明面板
        JPanel explain_right=new JPanel();
        //初始化格式布局
        explain_left.setLayout(new GridLayout(4,1));
        explain_right.setLayout(new GridLayout(3,1));
        //在左说明面板,添加说明文字
        explain_left.add(new JLabel("按空格键,方块变形"));
        explain_left.add(new JLabel("按左箭头,方块左移"));
        explain_left.add(new JLabel("按右箭头,方块右移"));
        explain_left.add(new JLabel("按下箭头,方块下落"));
        //设置游戏标签的内容为红色字体
        Label1.setForeground(Color.RED);
        //将游戏状态和得分、提示添加到右面板上
        explain_right.add(label);
        explain_right.add(Label);
        explain_right.add(Label1);
        //将左说明面板添加到窗口左侧
        this.add(explain_left,BorderLayout.WEST);
        //将右说明面板添加到窗口右侧
        this.add(explain_right,BorderLayout.EAST);

    }
    //开始游戏的方法
    public void game_begin() throws Exception {
        while (true){
            //判断游戏是否结束
            if(!isrunning){
                break;
            }
            //进行游戏
            game_run();
        }
        //在标签位置显示游戏结束
        Label1.setText("游戏状态:游戏结束!");
    }
    //随机生成下落方块形状的方法
    public void ranRect(){
        Random random=new Random();

      rect=allRect[random.nextInt(22)];
    }
    //游戏运行的方法
    public void game_run() throws Exception {
        ranRect();
        //方块下落位置
        x=0;
        y=5;
        for (int i = 0; i < game_x; i++) {
            Thread.sleep(time);
            if (game_pause) {
                i--;
            } else {
            //判断方块是否可以下落
            if (!canFall(x, y)) {
                //将data变成1,表示有方块占用
                changData(x, y);
                //循环遍历4层,看是否有行可以消除
                for (int j = x; j < x + 4; j++) {
                    int sum = 0;
                    for (int k = 1; k <= (game_y - 2); k++) {
                        if (data[j][k] == 1) {
                            sum++;
                        }
                    }
                    //判断是否有一行可以被消除
                    if (sum == (game_y - 2)) {
                        //消除J这一行
                        removeRow(j);
                    }
                }
                //判断游戏是否失败
                for (int j = 1; j < (game_y - 2); j++) {
                    if (data[3][j] == 1) {
                        isrunning = false;
                        break;
                    }
                }
                break;//方块无法下落,我们应该重新生成一个方块,并重新遍历26层
            } else {
                //方块可以下落,层数加一
                x++;
                //方块下落一格
                fall(x, y);
            }
        }
        }
    }
    //判断方块是否可以继续下落的方法
    public boolean canFall(int m,int n){
        //定义一个变量
        int temp=0x8000;
        //遍历4*4方格
        for (int i = 0; i <4 ; i++) {
            for (int j = 0; j < 4; j++) {
                if((temp & rect)!=0){
                    //判断该位置的下一行是否有方块
                    if(data[m+1][n]==1){
                        return false;
                    }
                }
                n++;
                temp >>=1;//右移一位
            }
            m++;
            n=n-4;//让n回归首列
        }
        //循环结束,可以下落
        return true;
    }
    //改变不可下降的方块对应的区域的值的方法
    public void changData(int m,int n){
        //定义一个变量
        int temp=0x8000;
        //遍历整个4*4的方块
        for (int i = 0; i < 4; i++) {
            for (int j = 0; j < 4; j++) {
                if((temp & rect)!=0){
                    data[m][n]=1;
                }
                n++;
                temp >>=1;//右移一位
            }
            m++;
            n=n-4;
        }
    }
    //移除某一行的所有方块,并让上面的方块掉落的方法
    public void removeRow(int row){
        int temp=100;
        for (int i = row; i >=1 ; i--) {
            for (int j = 1; j <=(game_y-2) ; j++) {
                //进行覆盖
                data[i][j]=data[i-1][j];

            }
        }
        //刷新游戏区域
        reflesh(row);
        //方块加速
        if(time>temp){
            time-=temp;
        }
        //每消除一行,得分加100
        score+=temp;
        //显示变化后的分数
        Label.setText("游戏得分:"+score);
    }
    //刷新移除某一行后的游戏界面的方法
    public void reflesh(int row){
        //遍历row上面的游戏区域
        for (int i = row; i >=1 ; i--) {
            for (int j = 1; j <=(game_y-2) ; j++) {
                if(data[i][j]==1){//如果是方块,将方块设置为蓝色
                    text[i][j].setBackground(Color.BLUE);
                }else{//如果不是方块,说明是游戏的背景区域
                    text[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;
        for (int i = 0; i < 4; i++) {
            for (int j = 0; j < 4; j++) {
                if ((temp & rect) != 0) {
                    text[m][n].setBackground(Color.WHITE);//将其设置成背景颜色,相当于消除
                }
                n++;
                temp >>=1;//右移一位
            }
            m++;
            n=n-4;
        }
    }
    //重新绘制掉落后的方块的方法
    public void draw(int m,int n){
        //定义一个变量
        int temp=0x8000;
        for (int i = 0; i < 4; i++) {
            for (int j = 0; j < 4; j++) {
                if((temp & rect)!=0){
                    text[m][n].setBackground(Color.BLUE);//设置成之前的方块颜色
                }
                n++;
                temp >>=1;//右移一位
            }
            m++;
            n=n-4;
        }
    }

    @Override
    public void keyTyped(KeyEvent e) {
        //控制游戏暂停
        if(e.getKeyChar()=='s'){//如果按下s,则游戏暂停
            //判断游戏是否结束
            if(!isrunning){
                return;
            }
            pause++;
            //判断是按下一次,暂停游戏
            if(pause==1){
                game_pause=true;
                Label1.setText("游戏状态:暂停中!");
            }
            //判断是按下两次,继续游戏
            if(pause==2){
                game_pause=false;
                pause=0;//重置暂停次数
                Label1.setText("游戏状态:正在游戏中!");
            }
        }

        //控制方块进行变形
        if(e.getKeyChar()==KeyEvent.VK_SPACE){
            //判断游戏是否结束
            if(!isrunning){
                return;
            }
            //判断游戏是否暂停
            if(game_pause){
                return;
            }
            //定义变量,存储目前方块的索引
            int old;
            for (old = 0; old < allRect.length; old++) {
                //判断是否是当前方块
                if(rect==allRect[old]){
                    break;
                }
            }
            //定义变量,存储变形后的方块
            int next;
            //判断是完整方块
            if(old==0||old==7||old==8||old==9){
                return;
            }
            //清除当前方块
            clear(x,y);
            if(old==1||old==2){
                next=allRect[old==1?2:1];
                //如果可以变形
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old>=3&&old<=6){
                next=allRect[old+1>6?3:old+1];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old==10||old==11){
                next=allRect[old==10?11:10];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old==12||old==13){
                next=allRect[old==12?13:12];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old>=14&&old<=17){
                next=allRect[old+1>17?14:old+1];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old==18||old==19){
                next=allRect[old==18?19:18];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            if(old==20||old==21){
                next=allRect[old==20?21:20];
                if(canTurn(next,x,y)){
                    rect=next;
                }
            }
            //重新绘制变形后的方块
            draw(x,y);
        }
    }
    public boolean canTurn(int a,int m,int n){
        //创建变量
        int temp=0x8000;
        //遍历整个方块
        for (int i = 0; i < 4; i++) {
            for (int j = 0; j < 4; j++) {
                if ((temp & rect) != 0) {
                    if(data[m][n]==1){
                        return false;
                    }
                }
                n++;
                temp >>=1;
            }
            m++;
            n=n-4;
        }
        //可以变形
        return true;
    }

    @Override
    public void keyPressed(KeyEvent e) {
        //方块进行左移
        if(e.getKeyCode()==37){//左箭头对应的数值为37
            //判断游戏是否结束
            if(!isrunning){
                return;
            }
            //判断游戏是否暂停
            if(game_pause){
                return;
            }
            //方块是否碰到左墙壁
            if(y<=1){
                return;
            }
            //方块的左边是否有方块
            int temp=0x8000;
            for (int i = x; i <x+4 ; i++) {
                for (int j = y; j <y+4 ; j++) {
                    if((temp & rect)!=0){
                        if(data[i][j-1]==1){
                            return;
                        }
                    }
                    temp >>=1;
                }
            }
            //清除目前方块
            clear(x,y);
            y--;//向左移动
            draw(x,y);//重新绘制出向左移动后的方块
        }
        //方块进行右移
        if(e.getKeyCode()==39) {//右箭头对应的数值为39
            //判断游戏是否结束
            if(!isrunning){
                return;
            }
            //判断游戏是否暂停
            if(game_pause){
                return;
            }
            int temp=0x8000;
            int m=x;
            int n=y;
            //存储最右边的坐标值
            int num=1;
            for (int i = 0; i < 4; i++) {
                for (int j = 0; j < 4; j++) {
                    if ((temp & rect) != 0) {
                        if(n>num){
                            num=n;
                        }
                    }
                    n++;
                    temp >>=1;
                }
                m++;
                n=n-4;
            }
            //判断是否碰到右墙壁
            if(num>=(game_y-2)){
                return;
            }
            //判断方块右移途中是否碰到其他方块
            temp=0x8000;
            for (int i = x; i <x+4 ; i++) {
                for (int j = y; j < y + 4; j++) {
                    if ((temp & rect) != 0) {
                        if(data[i][j+1]==1){
                            return;
                        }
                    }
                    temp>>=1;
                }
            }
            //清除当前方块
            clear(x,y);
            y++;//右移一位
            draw(x,y);//重新绘制出向右移动后的方块
        }
        //方块进行下落
        if(e.getKeyCode()==40) {//下箭头对应的数值为40
            //判断游戏是否结束
            if (!isrunning) {
                return;
            }
            //判断游戏是否暂停
            if(game_pause){
                return;
            }
            //判断方块是否可以下落
            if(!canFall(x,y)){
                return;
            }
            //清除当前方块
            clear(x,y);
            //改变方块坐标
            x++;
            draw(x,y);//重新绘制出向右移动后的方块
        }
    }

    @Override
    public void keyReleased(KeyEvent e) {

    }
}

其中的方块数组中的数字意义:

运行后结果:

到此这篇关于Java实现俄罗斯方块的源码分享的文章就介绍到这了,更多相关Java俄罗斯方块内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java实现俄罗斯方块游戏的示例代码

    目录 引言 效果图 实现思路 代码实现 创建窗口 画布1 创建菜单及菜单选项 绘制游戏区域 画布2 画布2绘制一个小方块 创建图形 创建模型类 模型旋转变形 方块累计 方块消除和积分 加入自动向下线程,并启动 引言 俄罗斯方块,相信很多80.90后的小伙伴都玩过,也是当年非常火的游戏,当年读中学的时候,有一个同学有这个游戏机,大家都很喜欢玩,这个游戏给当时的我们带来了很多欢乐,时光飞逝,感慨颇多! 人终归是要长大的,回忆再美好,日子也一去不复返了,以前我们只会玩游戏,心里想自己能做一个出来多牛逼

  • java实现俄罗斯方块小程序

    这是java课最后做的课程设计,由于java是初学的,所以做的时候有参考一些技术大牛的博客,在此表示感谢. 发在这里跟大家交流学习一下. 如需要完整工程文件.说明文档以及可运行jar文件,下载地址:点击打开链接 RussianBlocksGame.java package RussiaBlocksGame; import java.awt.*; import java.awt.event.*; import javax.swing.*; import javax.swing.border.Bor

  • Java实现简易俄罗斯方块

    本文实例为大家分享了Java实现简易俄罗斯方块的具体代码,供大家参考,具体内容如下 一.将对象抽象为类 首先考虑俄罗斯方块游戏中含有哪些具体的对象,对象中含有哪些具体属性和方法,然后用代码来实现. 建立如下类: Cell类:代表最小的方格单位,构成7种图形的最基本图形. 含有row(行号),col(列号),image(对应的图片)属性, 含有left(左移),right(右移),drop(下落)方法. Tetromino类:代表由4个最小方格构成的7种图形的合集. 含有cells(四个方块)属性

  • Java实现俄罗斯方块小游戏源码

    本文实例为大家分享了Java实现俄罗斯方块小游戏的具体代码,供大家参考,具体内容如下 一.最终效果 二.功能需求 1. 在二维平面里面用各种随机产生的方块堆积木,每满一行消去一行,当达到顶部时,游戏结束.2. 通过方向键来控制方块转动,左移,右移和直落.3. 方块下落统一设置蓝色,接触底部变粉色.4. 计算分数,分数是由方块的类型决定的,每堆积一个方块就把分数累加到总分中.5. 游戏有开始.重新开始.降低提高级数(速度).暂停.退出 三.程序实现 这个是最基础的方块素材 package 俄罗斯方

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

    本文实例为大家分享了Java实现俄罗斯方块游戏的具体代码,供大家参考,具体内容如下 游戏页面效果如下: 俄罗斯方块游戏本身的逻辑: 俄罗斯方块游戏的逻辑是比较简单的.它就类似于堆砌房子一样,各种各样的方地形状是不同的.但是,俄罗斯方块游戏的界面被等均的分为若干行和若干列,因此方块的本质就是占用了多少个单元. 首先来考虑一下数据的问题.对于界面来说,需要一个二维的 int 型数组,它保存着那些地方应该有着色,哪些没有:然后是方块本身,尽管它们的形状不统一,但是它们可以用一个4X4比例的方块所包围,

  • java代码实现俄罗斯方块

    本文实例为大家分享了java实现俄罗斯方块的具体代码,供大家参考,具体内容如下 俄罗斯方块设计思想 俄罗斯方块都从小玩到大吧,什么规则大家都知道了吧,以前感觉那玩意贼好玩,但是就是老赢不了,现在学会了自己写一个天天练! 键盘操作: 左键:左移: 右键:右移: 上键:变换造型 下键:加速下掉(没毛病吧,没有继续整) 任意一行的方块满格,这一行就消除,消除一行方块得10分,目前小主我还没有设置关卡,各位喜欢的宝宝们可以自己设置关卡哦: 那么那些方块的造型到底从哪里来的呢,那就是我们自己设计的,常见的

  • Java实现俄罗斯方块的源码分享

    本文实现的功能有: 1. 初始化游戏窗口 2.初始化游戏的界面 3.初始化游戏的说明面板 4.随机生成下落方块 5.方块下落速度变化 6.判断方块是否可以下落 7.移除某一行方块上面的方块后让上面的方块掉落 8.刷新移除某一行方块后的界面 9.清除方块 10.绘制方块 11.键盘控制方块的移动.变形和快速下落 12.游戏的暂停功能 三小时纯手工打造,具体实现代码: import javax.swing.*; import java.awt.*; import java.awt.event.Key

  • Java编程删除链表中重复的节点问题解决思路及源码分享

    一. 题目 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 二. 例子 输入链表:1->2->3->3->4->4->5 处理后为:1->2->5 三. 思路 个人感觉这题关键是注意指针的指向,可以定义一个first对象(值为-1,主要用于返回操作后的链表),first.next指向head,定义一个last同样指向first(主要用于操作记录要删除节点的前一个节点),定义一个p指向head,指向当前节点.

  • Java写入写出Excel操作源码分享

    这两天帮老师做一个数据库,将所有实验交易的数据导入到数据库中,但是不想天天在实验室里面待着,气氛太压抑,就想着先把数据读进EXCEL中,哪天带到实验室导进去 数据原来是这样的,不同的实验有一个专门的文件夹,实验名的文件夹下有不同班级的文件夹,班级文件夹下有该班级日期文件夹,存储的是不同时间下该班做实验的数据EXCEL,原来的EXCEL中没有班级和时间,现在需要通过读取EXCEL名以及班级名来将该信息作为一列,加入到EXCEL中. 下面是源代码,嘿嘿,顺便还做了一个可视化窗口. 类ExcelRea

  • Java实时获取基金收益项目源码分享

    本文章向大家介绍JAVA爬取天天基金网数据,主要包括JAVA爬取天天基金网数据使用实例.应用技巧.基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下. 天天基金网网址:http://quote.eastmoney.com/center/gridlist.html#fund_lof Java爬虫实时获取基金收益历史记录代码: 首先要自己定义几个参数:基金编码,页数,每页显示条数 开始时间结束时间等 (我这直接写的静态方法使用的 大家可以改成Test方法自行进行测试) /**

  • Android 用Time和Calendar获取系统当前时间源码分享(年月日时分秒周几)

    概述 用Time和Calendar获取系统当前时间(年月日时分秒周几) 效果图 源码: import android.app.Activity; import android.os.Bundle; import android.text.format.Time; import android.view.View; import android.widget.RelativeLayout; import android.widget.TextView; import java.util.Calen

  • Spring Boot 员工管理系统超详细教程(源码分享)

    员工管理系统 1.准备工作 资料下载 内含源码 + 笔记 + web素材 源码下载地址: http://xiazai.jb51.net/202105/yuanma/javaguanli_jb51.rar 笔记 素材 源码 1.1.导入资源 将文件夹中的静态资源导入idea中 位置如下 1.2.编写pojo层 员工表 //员工表 @Data @NoArgsConstructor public class Employee { private Integer id; private String l

  • 14 个Python小游戏 源码分享

    目录 1.吃金币 2.打乒乓 3.滑雪 4.并夕夕版飞机大战 5.打地鼠 6.小恐龙 7.消消乐 8.俄罗斯方块 9.贪吃蛇 10.24点小游戏 11.平衡木 12.外星人入侵 13.贪心鸟 14.井字棋888'' 1.吃金币 源码分享: import os import cfg import sys import pygame import random from modules import * '''游戏初始化''' def initGame(): # 初始化pygame, 设置展示窗口

  • Java Swing实现扫雷源码

    本文实例为大家分享了Java Swing实现扫雷源码的具体代码,供大家参考,具体内容如下 先来看下效果 运行时只需要创建一个GameWindow的对象即可,可使用有参构造函数自定义雷区行列数及炸弹个数,也可使用无参构造函数设置默认值(小旗和炸弹的图标自己去找吧,我就不在这里放了) import javax.swing.*; import java.awt.*; import java.awt.event.*; import java.util.LinkedList; public class G

  • java.lang.Void类源码解析

    在一次源码查看ThreadGroup的时候,看到一段代码,为以下: /* * @throws NullPointerException if the parent argument is {@code null} * @throws SecurityException if the current thread cannot create a * thread in the specified thread group. */ private static Void checkParentAcc

  • WPF实现类似360安全卫士界面的程序源码分享

    下面通过图文并茂的方式给大家介绍WPF实现类似360安全卫士界面的程序源码分享,点击此处下载源码哦. 以前学习Windows Form编程的时候,总感觉自己做的界面很丑,看到360安全卫士.迅雷等软件的UI设计都非常美观,心里总是憧憬着要是自己能实现这样的UI效果该多好!!!另一个困扰我的问题是,这个UI皮肤是如何用技术实现的呢?!虽然好多年过去了,但心里的憧憬和疑惑一直没有消失,而且越来越强烈.在日常的工作和学习中,自己在网上也经常留意类似的技术或者文章.最近在学习WPF的过程中,看到网上也有

随机推荐