热门标签 | HotTags
当前位置:  开发笔记 > 运维 > 正文

Java游戏俄罗斯方块的实现实例

这篇文章主要介绍了Java游戏俄罗斯方块的实现实例的相关资料,这里实现简单的俄罗斯方块帮助大家学习理解基础知识,需要的朋友可以参考下

Java游戏俄罗斯方块的实现实例

         java小游戏主要理解应用java Swing,awt等基础组件的知识,通过本例应当掌握面向对象的知识。

实现代码:

package cn.hncu.games;

import java.awt.Color;
import java.awt.Font;
import java.awt.Graphics;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.event.KeyAdapter;
import java.awt.event.KeyEvent;

import javax.swing.JFrame;
import javax.swing.JMenu;
import javax.swing.JMenuBar;
import javax.swing.JMenuItem;
import javax.swing.JOptionPane;
import javax.swing.JPanel;
import javax.swing.Timer;

public class Tetris extends JFrame {
  private TetrisPanel tp;

  public Tetris() {
    // 添加菜单,这里只是意思一下,留待大家自己做更详细的
    // 菜单条
    JMenuBar menubar = new JMenuBar();
    setJMenuBar(menubar);
    // 菜单
    JMenu menuGame = new JMenu("游戏");
    menubar.add(menuGame);
    // 菜单项
    JMenuItem mi1 = new JMenuItem("新游戏");
    mi1.setActionCommand("new");
    JMenuItem mi2 = new JMenuItem("暂停");
    mi2.setActionCommand("pause");
    JMenuItem mi3 = new JMenuItem("继续");
    mi3.setActionCommand("continue");
    JMenuItem mi4 = new JMenuItem("退出");
    mi4.setActionCommand("exit");

    menuGame.add(mi1);
    menuGame.add(mi2);
    menuGame.add(mi3);
    menuGame.add(mi4);

    //菜单项监听
    MenuListener menuListener = new MenuListener();
    mi1.addActionListener(menuListener);
    mi2.addActionListener(menuListener);
    mi3.addActionListener(menuListener);
    mi4.addActionListener(menuListener);

    // 版本菜单
    JMenu menuHelp = new JMenu("帮助");
    menubar.add(menuHelp);
    menuHelp.add("版本所有@湖南城院 QQ:666688888");


    setLocation(700, 200);
    setDefaultCloseOperation(EXIT_ON_CLOSE);
    setSize(220, 275);
    setResizable(false);
    tp = new TetrisPanel();
    getContentPane().add(tp);

    // 让整个画布添加键盘监听
    // tp.addKeyListener(tp.listener);//不行,画布不方便获得键盘焦点
    this.addKeyListener(tp.listener); // 让框架来监听键盘
  }

  public static void main(String[] args) {
    Tetris te = new Tetris();
    te.setVisible(true);
  }

  class MenuListener implements ActionListener {
    @Override
    public void actionPerformed(ActionEvent e) {
      if(e.getActionCommand().equalsIgnoreCase("new")){
        getContentPane().remove(tp);
        tp = new TetrisPanel();
        getContentPane().add(tp);
        getContentPane().validate();//校验当前容器,有刷新功能       
      }else if(e.getActionCommand().equalsIgnoreCase("pause")){
        tp.getTimer().stop();
      }else if(e.getActionCommand().equalsIgnoreCase("continue")){
        tp.getTimer().restart();
      }else if(e.getActionCommand().equalsIgnoreCase("exit")){
        System.exit(0);
      }
    }
  }
}

class TetrisPanel extends JPanel {
  private int map[][] = new int[13][23];// map[列号][行号]。真正的方块区是:21行*10列。边框(2列,1行)

  // 方块的形状:
  // 第一维代表方块类型(包括7种:S、Z、L、J、I、O、T)
  // 第二维代表旋转次数
  // 第三四维代表方块矩阵
  // shapes[type][turnState][i] i--> block[i/4][i%4]
  int shapes[][][] = new int[][][] {
  /*
   * 模板 { {0,0,0,0,0,0,0,0, 0,0,0,0, 0,0,0,0}, {0,0,0,0,0,0,0,0, 0,0,0,0,
   * 0,0,0,0}, {0,0,0,0,0,0,0,0, 0,0,0,0, 0,0,0,0}, {0,0,0,0,0,0,0,0, 0,0,0,0,
   * 0,0,0,0} }
   */
      // I (※把版本1中的横条从第1行换到第2行)
      { { 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0 },
          { 0, 0, 0, 0, 1, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0 } },
      // S
      { { 0, 0, 1, 1, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0 },
          { 0, 0, 1, 1, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0 } },
      // Z 第3行: shapes[2][2][]
      { { 1, 1, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 1, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0 } },
      // J
      { { 0, 1, 0, 0, 0, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0 },
          { 1, 0, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 } },
      // O
      { { 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 } },
      // L
      { { 1, 0, 0, 0, 1, 0, 0, 0, 1, 1, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 1, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 0, 0, 0, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
          { 0, 0, 1, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 } },
      // T
      { { 0, 1, 0, 0, 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 1, 1, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0 },
          { 1, 1, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
          { 0, 1, 0, 0, 0, 1, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0 } } };

  private int type;
  private int turnState;
  private int x, y;// 当前块的位置---左上角的坐标
  private int score = 0;
  private Timer timer = null;
  private int delay = 1000;

  TimerListener listener = null;

  public TetrisPanel() {
    newGame();
    nextBlock();

    listener = new TimerListener();
    timer = new Timer(delay, listener);
    timer.start();

  }

  private void newGame() {
    // 初始化游戏地图
    for (int i = 0; i <12; i++) {
      for (int j = 0; j <21; j++) {
        if (i == 0 || i == 11) {// 边框
          map[i][j] = 3;
        } else {
          map[i][j] = 0;
        }
      }
      map[i][21] = 3;
    }
    score = 0;
  }

  private void nextBlock() {
    type = (int) (Math.random() * 1000) % 7; // type=5;
    turnState = (int) (Math.random() * 1000) % 4; // turnState=3;
    x = 4;
    y = 0;
    if (crash(x, y, type, turnState) == 0) {
      timer.stop();
      int op = JOptionPane.showConfirmDialog(null,
          "Game Over!....笨蛋,敢再来一局吗&#63;!");
      if (op == JOptionPane.YES_OPTION) {
        newGame();
      } else if (op == JOptionPane.NO_OPTION) {
        System.exit(0);
      }
    }
  }

  private void down() {
    if (crash(x, y + 1, type, turnState) == 0) {// 判断当前块往下落一格后是否和地图存在填充块完全重合---注意实参:y+1
      add(x, y, type, turnState);// 把该块加到地图---形成堆积块
      nextBlock();
    } else {
      y++;
    }
    repaint();
  }

  private void left() {
    if (x >= 0) {
      x -= crash(x - 1, y, type, turnState);
    }
    repaint();
  }

  private void right() {
    if (x <8) {
      x += crash(x + 1, y, type, turnState);
    }
    repaint();
  }

  private void turn() {
    if (crash(x, y, type, (turnState + 1) % 4) == 1) {
      turnState = (turnState + 1) % 4;
    }
    repaint();
  }

  // 让一个块堆积,其实是把当前块中的填充块信息记录到map[][]中
  private void add(int x, int y, int type, int turnState) {
    for (int a = 0; a <4; a++) {
      for (int b = 0; b <4; b++) {
        if (shapes[type][turnState][a * 4 + b] == 1) {
          map[x + b + 1][y + a] = 1;
        }
      }
    }
    tryDelLine();
  }

  // 消块
  private void tryDelLine() {
    // 从上往下,一行行依次遍历,如果某一行的map[i][j]值全是1,则把这一行消掉---上一行往下落
    for (int b = 0; b <21; b++) {
      int c = 1;
      for (int a = 0; a <12; a++) {
        c &= map[a][b];
      }
      if (c == 1) {// 全是1--下落一行
        score += 10;
        for (int d = b; d > 0; d--) {
          for (int e = 0; e <11; e++) {
            map[e][d] = map[e][d - 1];
          }
        }

        // 更改游戏的难度(加快下落速度)
        delay /= 2;
        timer.setDelay(delay);
      }

    }

  }

  private int crash(int x, int y, int blockType, int turnState) {
    for (int a = 0; a <4; a++) {
      for (int b = 0; b <4; b++) {
        if ((shapes[blockType][turnState][a * 4 + b] & map[x + b + 1][y
            + a]) == 1) {// 和填充块或框架重合,都算碰撞
          return 0; // 碰撞了---方块的填充块和地图中的填充块完全重合
        }
      }
    }
    return 1;// 没有碰撞
  }

  // 表现层
  @Override
  public void paint(Graphics g) {
    super.paint(g);// 清除残影

    // 画当前块
    for (int j = 0; j <16; j++) {
      if (shapes[type][turnState][j] == 1) {
        g.setColor(Color.green);
        g.fillRect((j % 4 + x + 1) * 10, (j / 4 + y) * 10, 10, 10);
      }
    }

    /*
     * for(int a=0;a<4;a++){ for(int b=0;b<4;b++){
     * if(shapes[type][turnState][a*4+b]==1){ g.fillRect((b+x+1)*10,
     * (a+y)*10, 10, 10); } } }
     */

    // 画地图(整个游戏的方块区和边框)
    for (int i = 0; i <12; i++) {
      for (int j = 0; j <22; j++) {
        if (map[i][j] == 1) {
          g.setColor(Color.red);
          g.fillRect(i * 10, j * 10, 10, 10);// 填充

          g.setColor(Color.yellow);
          g.drawRect(i * 10, j * 10, 10, 10);// 格线
        } else if (map[i][j] == 3) {
          g.setColor(Color.red);
          g.drawRect(i * 10, j * 10, 10, 10);
        }
      }
    }

    // 显示分数,同时为版面美观,在界面上再加点东西
    // 画方块区右侧部分
    g.setColor(Color.blue);
    g.setFont(new Font("aa", Font.BOLD, 18));
    g.drawString("score=" + score, 130, 20);

    g.setFont(new Font("aa", Font.PLAIN, 13));
    g.drawString("拒绝盗版游戏", 130, 70);
    g.drawString("注意自我保护", 130, 90);
    g.drawString("谨防受骗上当。", 125, 110);
    g.drawString("适度游戏益脑,", 125, 130);
    g.drawString("沉迷游戏伤身。", 125, 150);
    g.drawString("合理安排时间,", 125, 170);
    g.drawString("享受健康生活。", 125, 190);

  }

  class TimerListener extends KeyAdapter implements ActionListener {
    @Override
    public void actionPerformed(ActionEvent e) {
      down();
    }

    @Override
    public void keyPressed(KeyEvent e) {
      // System.out.println("aaaaa");
      switch (e.getKeyCode()) {
      case KeyEvent.VK_DOWN:
        down();
        break;
      case KeyEvent.VK_LEFT:
        left();
        break;
      case KeyEvent.VK_RIGHT:
        right();
        break;
      case KeyEvent.VK_UP:
        turn();
      }
    }

  }

  public Timer getTimer() {
    return timer;
  }
}

以上就是Java 俄罗斯方块的详解,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


推荐阅读
  • docker增加restart=always, docker重启后自动启动容器的方法
    本文介绍了在运行docker容器时如何添加参数来保证每次docker服务重启后容器也自动重启的方法,以及如何使用命令来更新已启动的容器。 ... [详细]
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • 本文介绍了在Docker容器技术中限制容器对CPU的使用的方法,包括使用-c参数设置容器的内存限额,以及通过设置工作线程数量来充分利用CPU资源。同时,还介绍了容器权重分配的情况,以及如何通过top命令查看容器在CPU资源紧张情况下的使用情况。 ... [详细]
  • 集合的遍历方式及其局限性
    本文介绍了Java中集合的遍历方式,重点介绍了for-each语句的用法和优势。同时指出了for-each语句无法引用数组或集合的索引的局限性。通过示例代码展示了for-each语句的使用方法,并提供了改写为for语句版本的方法。 ... [详细]
  • Python SQLAlchemy库的使用方法详解
    本文详细介绍了Python中使用SQLAlchemy库的方法。首先对SQLAlchemy进行了简介,包括其定义、适用的数据库类型等。然后讨论了SQLAlchemy提供的两种主要使用模式,即SQL表达式语言和ORM。针对不同的需求,给出了选择哪种模式的建议。最后,介绍了连接数据库的方法,包括创建SQLAlchemy引擎和执行SQL语句的接口。 ... [详细]
  • position属性absolute与relative的区别和用法详解
    本文详细解读了CSS中的position属性absolute和relative的区别和用法。通过解释绝对定位和相对定位的含义,以及配合TOP、RIGHT、BOTTOM、LEFT进行定位的方式,说明了它们的特性和能够实现的效果。同时指出了在网页居中时使用Absolute可能会出错的原因,即以浏览器左上角为原始点进行定位,不会随着分辨率的变化而变化位置。最后总结了一些使用这两个属性的技巧。 ... [详细]
  • 开发笔记:Docker 上安装启动 MySQL
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了Docker上安装启动MySQL相关的知识,希望对你有一定的参考价值。 ... [详细]
  • Oracle优化新常态的五大禁止及其性能隐患
    本文介绍了Oracle优化新常态中的五大禁止措施,包括禁止外键、禁止视图、禁止触发器、禁止存储过程和禁止JOB,并分析了这些禁止措施可能带来的性能隐患。文章还讨论了这些禁止措施在C/S架构和B/S架构中的不同应用情况,并提出了解决方案。 ... [详细]
  • Spring常用注解(绝对经典),全靠这份Java知识点PDF大全
    本文介绍了Spring常用注解和注入bean的注解,包括@Bean、@Autowired、@Inject等,同时提供了一个Java知识点PDF大全的资源链接。其中详细介绍了ColorFactoryBean的使用,以及@Autowired和@Inject的区别和用法。此外,还提到了@Required属性的配置和使用。 ... [详细]
  • 本文介绍了Java的公式汇总及相关知识,包括定义变量的语法格式、类型转换公式、三元表达式、定义新的实例的格式、引用类型的方法以及数组静态初始化等内容。希望对读者有一定的参考价值。 ... [详细]
  • 本文讨论了微软的STL容器类是否线程安全。根据MSDN的回答,STL容器类包括vector、deque、list、queue、stack、priority_queue、valarray、map、hash_map、multimap、hash_multimap、set、hash_set、multiset、hash_multiset、basic_string和bitset。对于单个对象来说,多个线程同时读取是安全的。但如果一个线程正在写入一个对象,那么所有的读写操作都需要进行同步。 ... [详细]
  • 本文介绍了一种图片处理应用,通过固定容器来实现缩略图的功能。该方法可以实现等比例缩略、扩容填充和裁剪等操作。详细的实现步骤和代码示例在正文中给出。 ... [详细]
  • C++语言入门:数组的基本知识和应用领域
    本文介绍了C++语言的基本知识和应用领域,包括C++语言与Python语言的区别、C++语言的结构化特点、关键字和控制语句的使用、运算符的种类和表达式的灵活性、各种数据类型的运算以及指针概念的引入。同时,还探讨了C++语言在代码效率方面的优势和与汇编语言的比较。对于想要学习C++语言的初学者来说,本文提供了一个简洁而全面的入门指南。 ... [详细]
  • 本文介绍了H5游戏性能优化和调试技巧,包括从问题表象出发进行优化、排除外部问题导致的卡顿、帧率设定、减少drawcall的方法、UI优化和图集渲染等八个理念。对于游戏程序员来说,解决游戏性能问题是一个关键的任务,本文提供了一些有用的参考价值。摘要长度为183字。 ... [详细]
  • 本文介绍了Python函数的定义与调用的方法,以及函数的作用,包括增强代码的可读性和重用性。文章详细解释了函数的定义与调用的语法和规则,以及函数的参数和返回值的用法。同时,还介绍了函数返回值的多种情况和多个值的返回方式。通过学习本文,读者可以更好地理解和使用Python函数,提高代码的可读性和重用性。 ... [详细]
author-avatar
玲玲0308baby
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有