Java学习-第一部分-第二阶段-项目实战:坦克大战【2】

坦克大战【2】

笔记目录:(https://www.cnblogs.com/wenjie2000/p/16378441.html)

线程-应用到坦克大战

坦克大战0.3版

陆游曾说:纸上得来总觉浅,绝知此事要躬行。前面我要已经了解java线程基本知识,现在我们来实际运用一下。
在坦克大战游戏(0.2版)基础上添加如下功能:当玩家按一下j键,就发射一颗子弹.

分析如何实现当用户按下J键,我们的坦克就发射一颗子弹.
思路

  1. 当发射一颗子弹后,就相当于启动一个线程
  2. Hero有子弹的对象,当按下J时,我们就启动一个发射行为(线程),让子弹不停的移动,形成一个射击的效果
  3. 我们MyPanel需要不停的重绘子弹,才能出现该效果.
  4. 当子弹移动到面板的边界时,就应该销毁(把启动的子弹的线程销毁)

具体代码如下

Tank

package tankgame03;

public class Tank {
    //坦克左上角坐标
    private int x;
    private int y;
    private int direct;//坦克方向 0上 1右 2下 3左

    public int getSpeed() {
        return speed;
    }

    public void setSpeed(int speed) {
        this.speed = speed;
    }

    private int speed=1;//坦克速度

    //上右下左移动方法
    public void moveUp(){
        y-=speed;
    }
    public void moveDown(){
        y+=speed;
    }
    public void moveLeft(){
        x-=speed;
    }
    public void moveRight(){
        x+=speed;
    }

    public int getDirect() {
        return direct;
    }

    public void setDirect(int direct) {
        this.direct = direct;
    }

    public Tank(int x, int y) {
        this.x = x;
        this.y = y;
    }

    public int getX() {
        return x;
    }

    public void setX(int x) {
        this.x = x;
    }

    public int getY() {
        return y;
    }

    public void setY(int y) {
        this.y = y;
    }
}

Hero

package tankgame03;

public class Hero extends Tank {
    Shot shot=null;//射击行为
    public Hero(int x, int y) {
        super(x, y);
    }
    //射击
    public void shotEnemyTank(){
        switch (getDirect()){
            case 0://上
                shot=new Shot(getX()+20,getY(),0);
                break;
            case 1://右边
                shot=new Shot(getX()+50,getY()+30,1);
                break;
            case 2:
                shot=new Shot(getX()+20,getY()+60,2);
                break;
            case 3:
                shot=new Shot(getX()-10,getY()+30,3);
                break;
        }
        new Thread(shot).start();
    }
}

EnemyTank

package tankgame03;

public class Hero extends Tank {
    Shot shot=null;//射击行为
    public Hero(int x, int y) {
        super(x, y);
    }
    //射击
    public void shotEnemyTank(){
        switch (getDirect()){
            case 0://上
                shot=new Shot(getX()+20,getY(),0);
                break;
            case 1://右边
                shot=new Shot(getX()+50,getY()+30,1);
                break;
            case 2:
                shot=new Shot(getX()+20,getY()+60,2);
                break;
            case 3:
                shot=new Shot(getX()-10,getY()+30,3);
                break;
        }
        new Thread(shot).start();
    }
}

Shot

package tankgame03;

public class Shot implements Runnable {
    private int x;
    private int y;
    private int direct;//方向 0上 1右 2下 3左
    private int speed = 10;//子弹速度
    boolean isAlive = true;

    public Shot(int x, int y, int direct) {
        this.x = x;
        this.y = y;
        this.direct = direct;
    }

    public int getX() {
        return x;
    }

    public int getY() {
        return y;
    }

    @Override
    public void run() {
        while (true) {
            //休眠
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //根据方向移动
            switch (direct) {
                case 0:
                    y -= speed;
                    break;
                case 1:
                    x += speed;
                    break;
                case 2:
                    y += speed;
                    break;
                case 3:
                    x -= speed;
                    break;
            }
            System.out.println("子弹坐标(" + x + "," + y + ")");
            if (x < 0 || x >= 1000 || y < 0 || y >= 750) {
                System.out.println("子弹线程退出");
                isAlive = false;
                break;
            }
        }
    }
}

MyPanel

package tankgame03;

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

////为了监听键盘事件,实现KeyListener
//为了让Panel不停的重绘子弹,需要将MyPanel实现Runnable ,当做一个线程使用
public class MyPanel extends JPanel implements KeyListener,Runnable {
    Hero hero = null;//定义我的坦克
    Vector<EnemyTank> enemyTanks = new Vector<>();//定义敌人坦克
    int enemyTankSize = 3;

    public MyPanel() {
        hero = new Hero(100, 100);//初始化自己的坦克位置
        for (int i = 0; i < enemyTankSize; i++) {//初始化敌人的坦克位置
            enemyTanks.add(new EnemyTank(100 * (i + 1), 0));
        }

//        hero.setSpeed(5);
    }

    @Override
    public void paint(Graphics g) {
        super.paint(g);
        g.fillRect(0, 0, 1000, 750);//黑色背景

        //画出坦克
        drawTank(hero.getX(), hero.getY(), g, hero.getDirect(), 0);

        //画出敌人坦克
        for (int i = 0; i < enemyTanks.s; i++) {
            //取出坦克
            EnemyTank enemyTank = enemyTanks.get(i);
            enemyTank.setDirect(2);
            drawTank(enemyTank.getX(), enemyTank.getY(), g, enemyTank.getDirect(), 1);
        }

        //画出hero发出的子弹
        if(hero.shot!=null&&hero.shot.isAlive==true){
            g.fill3DRect(hero.shot.getX()-1,hero.shot.getY()-1,3,3,false);
        }
    }

    /**
     * @param x      坦克左上角x坐标
     * @param y      坦克左上角y坐标
     * @param g      画笔
     * @param direct 坦克方向(上下左右)
     * @param type   坦克类型
     */
    public void drawTank(int x, int y, Graphics g, int direct, int type) {

        switch (type) {
            case 0://我们的坦克
                g.setColor(Color.cyan);
                break;
            case 1://敌人的坦克
                g.setColor(Color.yellow);
                break;
        }
        //根据坦克方向,绘制坦克
        switch (direct) {
            case 0://向上
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y);//画出炮筒
                break;
            case 1://右
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 50, y + 30);//画出炮筒
                break;
            case 2://下
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y + 60);//画出炮筒
                break;
            case 3://左
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x -10, y + 30);//画出炮筒
                break;
        }

    }

    @Override
    public void keyTyped(KeyEvent e) {

    }

    //处理WDSA键按下情况
    @Override
    public void keyPressed(KeyEvent e) {
        if (e.getKeyCode() == KeyEvent.VK_W) {
            //改变坦克方向
            hero.setDirect(0);
            //修改坦克坐标
            hero.moveUp();
        } else if (e.getKeyCode() == KeyEvent.VK_D) {
            hero.setDirect(1);
            hero.moveRight();
        } else if (e.getKeyCode() == KeyEvent.VK_S) {
            hero.setDirect(2);
            hero.moveDown();
        } else if (e.getKeyCode() == KeyEvent.VK_A) {
            hero.setDirect(3);
            hero.moveLeft();
        }

        if(e.getKeyCode()==KeyEvent.VK_J){
            hero.shotEnemyTank();
        }
//        this.repaint();
    }

    @Override
    public void keyReleased(KeyEvent e) {

    }

    @Override
    public void run() {
        while (true){
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            this.repaint();
        }
    }
}

HspTankGame03

package tankgame03;

import javax.swing.*;

public class HspTankGame03 extends JFrame {
    MyPanel mp=null;
    public static void main(String[] args) {
        new HspTankGame03();
    }

    public HspTankGame03() {
        mp=new MyPanel();
        new Thread(mp).start();
        this.add(mp);
        this.setSize(1000,750);
        this.addKeyListener(mp);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setVisible(true);
    }
}

坦克大战0.4版

增加功能[HspTankGame04.java]

  1. 让敌人的坦克也能够发射子弹(可以有多颗子弹)
  2. 当我方坦克击中敌人坦克时,敌人的坦克就消失,如果能做出爆炸效果更好.
  3. 让敌人的坦克也可以自由随机的上下左右移动
  4. 控制我方的坦克和敌人的坦克在规定的范围移动

√特别说明:

  1. 只要能实现就行,方法的好坏后面再说。
  2. 完成上面的任务,不会有没有讲过的知识点,这里主要是锻炼灵活运用技术点能力。
  3. 一定要自己先动脑筋想想,试着做做。再听老师的评讲才有意义,时间自己掌控(再次提醒,一定要先自己思考,再看代码收获会很大)

思路

让敌人的坦克也能够发射子弹(可以有多颗子弹)

  1. 在敌人坦克类,使用Vector保存多个Shot
  2. 当每创建一个敌人坦克对象,给该敌人坦克对象初始化一个shot对象,同时启动Shot
  3. 在绘制敌人坦克时,需要遍历敌人坦克对象Vector,绘制所有的子弹,当子弹isLive == false时,就从Vector移除

让敌人的坦克也可以自由随机的上下左右移动思路分析

  1. 因为要求敌人的坦克,可以自由移动,因此需要将敌人坦克当做线程使用
  2. 我们需要Enemy Tank implements Runnable
  3. 在run方法写上我们相应的业务代码.
  4. 在创建敌人坦克对象时,启动线程

以下为程序代码(爆炸效果图片需要加到编译文件的根目录中)
(1).png
(1).png
(2).png
(2).png
(3).png
(3).png

Tank

package tankgame04;

public class Tank {
    //坦克左上角坐标
    private int x;
    private int y;
    private int direct;//坦克方向 0上 1右 2下 3左

    public int getSpeed() {
        return speed;
    }

    public void setSpeed(int speed) {
        this.speed = speed;
    }

    private int speed=5;//坦克速度

    //上右下左移动方法
    public void moveUp(){
        if(y-speed>0){
            y-=speed;
        }
    }
    public void moveDown(){
        if (y+60+speed<750){
            y += speed;
        }
    }
    public void moveLeft(){
        if (x-10-speed>0){
            x -= speed;
        }
    }
    public void moveRight(){
        if(x-10+60+speed<1000){
            x+=speed;
        }
    }

    public int getDirect() {
        return direct;
    }

    public void setDirect(int direct) {
        this.direct = direct;
    }

    public Tank(int x, int y) {
        this.x = x;
        this.y = y;
    }

    public int getX() {
        return x;
    }

    public void setX(int x) {
        this.x = x;
    }

    public int getY() {
        return y;
    }

    public void setY(int y) {
        this.y = y;
    }
}

Hero

package tankgame04;
//玩家坦克
public class Hero extends Tank {
    Shot shot=null;//射击行为
    public Hero(int x, int y) {
        super(x, y);
    }
    //射击
    public void shotEnemyTank(){
        switch (getDirect()){
            case 0://上
                shot=new Shot(getX()+20,getY(),0);
                break;
            case 1://右边
                shot=new Shot(getX()+50,getY()+30,1);
                break;
            case 2:
                shot=new Shot(getX()+20,getY()+60,2);
                break;
            case 3:
                shot=new Shot(getX()-10,getY()+30,3);
                break;
        }
        new Thread(shot).start();
    }
}

EnemyTank

package tankgame04;

import java.util.Vector;
//敌方坦克
public class EnemyTank extends Tank implements Runnable {
    //在敌人坦克类,使用Vector保存多个Shot
    Vector<Shot> shots = new Vector<>();
    boolean isLive = true;

    public EnemyTank(int x, int y) {
        super(x, y);
    }

    @Override
    public void run() {
        while (true) {
            //根据坦克方向继续移动
            for (int i=0;i<30;i++){
                switch (getDirect()) {
                    case 0://向上
                        //让坦克保持一个方向,走30步
                        moveUp();
                        break;
                    case 1://向右
                        moveRight();
                        break;
                    case 2://向下
                        moveDown();
                        break;
                    case 3://向左
                        moveLeft();
                        break;
                }
                //休眠50毫秒
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            //然后随机的改变坦克方向 0-3
            setDirect((int) (Math.random() * 4));
            if(!isLive){
                break;
            }
        }
    }
}

Shot

package tankgame04;
//射击事件
public class Shot implements Runnable {
    private int x;
    private int y;
    private int direct;//方向 0上 1右 2下 3左
    private int speed = 10;//子弹速度
    boolean isAlive = true;

    public Shot(int x, int y, int direct) {
        this.x = x;
        this.y = y;
        this.direct = direct;
    }

    public int getX() {
        return x;
    }

    public int getY() {
        return y;
    }

    @Override
    public void run() {
        while (true) {
            //休眠
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //根据方向移动
            switch (direct) {
                case 0:
                    y -= speed;
                    break;
                case 1:
                    x += speed;
                    break;
                case 2:
                    y += speed;
                    break;
                case 3:
                    x -= speed;
                    break;
            }
//            System.out.println("子弹坐标(" + x + "," + y + ")");
            //当子弹移动到面板的边界时,就应该销毁(把启动的子弹的线程销毁)
            // 当子弹碰到敌人坦克时,也应该结束线程
            if (x < 0 || x >= 1000 || y < 0 || y >= 750 || isAlive == false) {
//                System.out.println("子弹线程退出");
                isAlive = false;
                break;
            }
        }
    }
}

Bomb

package tankgame04;
//爆炸事件

public class Bomb {
    int x, y;//炸弹的坐标
    int life = 9;//炸弹的生命周期
    boolean isLive = true;//是否还存活

    public Bomb(int x, int y) {
        this.x = x;
        this.y = y;
    }

    //减少爆炸效果的生命值
    public void lifeDown() {//配合出现图片的爆炸效果
        if (life > 0) {
            life--;
        } else {
            isLive = false;
        }
    }
}

MyPanel

package tankgame04;
//面板
import javax.swing.*;
import java.awt.*;
import java.awt.event.KeyEvent;
import java.awt.event.KeyListener;
import java.util.Vector;

////为了监听键盘事件,实现KeyListener
//为了让Panel不停的重绘子弹,需要将MyPanel实现Runnable ,当做一个线程使用
public class MyPanel extends JPanel implements KeyListener, Runnable {
    Hero hero = null;//定义我的坦克
    Vector<EnemyTank> enemyTanks = new Vector<>();//定义敌人坦克
    //定义一个Vector ,用于存放多个爆炸效果
    //说明,当子弹击中坦克时,加入一个Bomb对象到bombs
    Vector<Bomb> bombs = new Vector<>();
    int enemyTankSize = 3;

    //定义三张炸弹图片,用于显示爆炸效果
    Image image1 = null;
    Image image2 = null;
    Image image3 = null;

    public MyPanel() {
        hero = new Hero(100, 100);//初始化自己的坦克位置
        for (int i = 0; i < enemyTankSize; i++) {//初始化敌人的坦克位置
            EnemyTank enemyTank = new EnemyTank(100 * (i + 1), 0);
            enemyTank.setDirect(2);//设置坦克方向
            //启动敌人坦克线程,让他们动起来
            new Thread(enemyTank).start();
            Shot shot = new Shot(enemyTank.getX() + 20, enemyTank.getY() + 60, 2);//创建敌人子弹
            enemyTank.shots.add(shot);//加入enemyTanks的Vector集合
            //启动shot对象
            new Thread(shot).start();//敌方子弹移动

            enemyTanks.add(enemyTank);//加入敌方坦克集合
        }
//        hero.setSpeed(5);
        image1 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(1).png"));
        image2 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(2).png"));
        image3 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(3).png"));

        //图片在被第一次使用时才真正加载
        // 所以会出现第一次爆炸的图程序找不到还没加载好图片导致这次爆炸效果缺失的情况。
        //因此要提前自定义爆炸一次
//        bombs.add(new Bomb(-100,-100));
    }

    @Override
    public void paint(Graphics g) {
        super.paint(g);

        g.fillRect(0, 0, 1000, 750);//黑色背景
        //画出hero坦克
        drawTank(hero.getX(), hero.getY(), g, hero.getDirect(), 0);

        //画出hero发出的子弹
        if (hero.shot != null && hero.shot.isAlive == true) {
            g.fill3DRect(hero.shot.getX() - 1, hero.shot.getY() - 1, 3, 3, false);
        }

        for (int i = 0; i < bombs.size(); i++) {
            //取出炸弹
            Bomb bomb = bombs.get(i);
            //根据当前这个bomb对象的life值去画出对应的图片
            if (bomb.life > 6) {
                g.drawImage(image1, bomb.x, bomb.y, 60, 60, this);
            } else if (bomb.life > 3) {
                g.drawImage(image2, bomb.x, bomb.y, 60, 60, this);
            } else {
                g.drawImage(image3, bomb.x, bomb.y, 60, 60, this);
            }
            //让这个炸弹的生命值减少
            bomb.lifeDown();
            //如果bomb life为0,就从bombs 的集合中删除
            if (bomb.life <= 0) {
                bombs.remove(bomb);
            }
        }

        //画出敌人坦克
        for (int i = 0; i < enemyTanks.size(); i++) {
            //取出坦克
            EnemyTank enemyTank = enemyTanks.get(i);
            if (enemyTank.isLive) {//判断敌方坦克是否存活
                drawTank(enemyTank.getX(), enemyTank.getY(), g, enemyTank.getDirect(), 1);
                //画出 enemyTank所有子弹
                for (int j = 0; j < enemyTank.shots.size(); j++) {
                    //取出子弹
                    Shot shot = enemyTank.shots.get(j);//绘制
                    if (shot.isAlive) { //isAlive = true
                        g.draw3DRect(shot.getX() - 1, shot.getY() - 1, 3, 3, false);
                    } else {
                        //从Vector移除
                        enemyTank.shots.remove(shot);
                    }
                }
            }
        }

    }

    /**
     * @param x      坦克左上角x坐标
     * @param y      坦克左上角y坐标
     * @param g      画笔
     * @param direct 坦克方向(上下左右)
     * @param type   坦克类型
     */
    public void drawTank(int x, int y, Graphics g, int direct, int type) {

        switch (type) {
            case 0://我们的坦克
                g.setColor(Color.cyan);
                break;
            case 1://敌人的坦克
                g.setColor(Color.yellow);
                break;
        }
        //根据坦克方向,绘制坦克
        switch (direct) {
            case 0://向上
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y);//画出炮筒
                break;
            case 1://右
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 50, y + 30);//画出炮筒
                break;
            case 2://下
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y + 60);//画出炮筒
                break;
            case 3://左
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x - 10, y + 30);//画出炮筒
                break;
        }

    }

    //编写方法,判断我方的子弹是否击中敌人坦克
    public void hitTank(Shot s, EnemyTank enemyTank) {
        switch (enemyTank.getDirect()) {
            case 0://上
            case 2://下
                if (s.getX() > enemyTank.getX() && s.getX() < enemyTank.getX() + 40 &&
                        s.getY() > enemyTank.getY() && s.getY() < enemyTank.getY() + 60) {
                    s.isAlive = false;//设置子弹为死亡状态
                    enemyTank.isLive = false;//设置该坦克为死亡状态
                    //创建Bomb对象,加入到bombs集合
                    Bomb bomb = new Bomb(enemyTank.getX()-10, enemyTank.getY());
                    bombs.add(bomb);
                    //当我的子弹击中敌人坦克后,将enemyTank 从lector拿掉
                    enemyTanks.remove(enemyTank);//移除该坦克
                }
                break;
            case 1:
            case 3:
                if (s.getX() > enemyTank.getX() - 10 && s.getX() < enemyTank.getX() + 50 &&
                        s.getY() > enemyTank.getY() + 10 && s.getY() < enemyTank.getY() + 50) {
                    s.isAlive = false;
                    enemyTank.isLive = false;
                }
                break;
        }
    }

    @Override
    public void keyTyped(KeyEvent e) {
    }

    //处理WDSA键按下情况
    @Override
    public void keyPressed(KeyEvent e) {
        if (e.getKeyCode() == KeyEvent.VK_W) {
            //改变坦克方向
            hero.setDirect(0);
            //修改坦克坐标
            hero.moveUp();
        } else if (e.getKeyCode() == KeyEvent.VK_D) {
            hero.setDirect(1);
            hero.moveRight();
        } else if (e.getKeyCode() == KeyEvent.VK_S) {
            hero.setDirect(2);
            hero.moveDown();
        } else if (e.getKeyCode() == KeyEvent.VK_A) {
            hero.setDirect(3);
            hero.moveLeft();
        }

        if (e.getKeyCode() == KeyEvent.VK_J) {
            hero.shotEnemyTank();
        }
//        this.repaint();
    }

    @Override
    public void keyReleased(KeyEvent e) {

    }

    @Override
    public void run() {
        while (true) {
            try {
                Thread.sleep(100);//每停0.1s执行一次
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            this.repaint();//刷新面板
            if (hero.shot != null && hero.shot.isAlive) {
                //遍历敌人所有坦克,实时判断坦克是否被击中
                for (int i = 0; i < enemyTanks.size(); i++) {
                    EnemyTank enemyTank = enemyTanks.get(i);
                    hitTank(hero.shot, enemyTank);
                }
            }
        }
    }
}

HspTankGame04

package tankgame04;
//窗体
import javax.swing.*;

public class HspTankGame04 extends JFrame {
    MyPanel mp=null;
    public static void main(String[] args) {
        new HspTankGame04();
    }

    public HspTankGame04() {
        mp=new MyPanel();
        new Thread(mp).start();//另外开一线程一直刷新面板内容和面板中一些需要同时执行的程序
        this.add(mp);
        this.setSize(1000,750);
        this.addKeyListener(mp);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setVisible(true);
    }
}

坦克大战0.5版

√增加功能 [HspTankGame05.java]

  1. 我方坦克在发射的子弹消亡后,才能发射新的子弹.=>扩展(发多颗子弹怎么办,改成面板最多只能存在五个自己的子弹)
  2. 让敌人坦克发射的子弹消亡后,可以再发射子弹
  3. 当敌人的坦克击中我方坦克时,我方坦克消失,并出现爆炸效果.

我方坦克在发射的子弹消亡后,才能发射新的子弹.=>扩展(发多颗子弹怎么办)
思路

  1. 在按下J键,我们判断当前hero对象的子弹,是否已经销毁
  2. 如果没有销毁,就不去触发shotEnemyTank
  3. 如果已经销毁,才去触发shotEnemyTank
  4. 如果要发射多颗子弹,就使用Vector保存

Tank

package tankgame05;
//坦克
public class Tank {
    //坦克左上角坐标
    private int x;
    private int y;
    private int direct;//坦克方向 0上 1右 2下 3左
    boolean isLive = true;//是否存活

    public int getSpeed() {
        return speed;
    }

    public void setSpeed(int speed) {
        this.speed = speed;
    }

    private int speed=2;//坦克速度

    //上右下左移动方法
    public void moveUp(){
        if(y-speed>0){
            y-=speed;
        }
    }
    public void moveDown(){
        if (y+60+speed<750){
            y += speed;
        }
    }
    public void moveLeft(){
        if (x-10-speed>0){
            x -= speed;
        }
    }
    public void moveRight(){
        if(x-10+60+speed<1000){
            x+=speed;
        }
    }

    public int getDirect() {
        return direct;
    }

    public void setDirect(int direct) {
        this.direct = direct;
    }

    public Tank(int x, int y) {
        this.x = x;
        this.y = y;
    }

    public int getX() {
        return x;
    }

    public void setX(int x) {
        this.x = x;
    }

    public int getY() {
        return y;
    }

    public void setY(int y) {
        this.y = y;
    }
}

Hero

package tankgame05;

import java.util.Vector;

//玩家坦克
public class Hero extends Tank {
    Shot shot = null;//射击行为
    //发射多个子弹
    Vector<Shot> shots = new Vector<>();

    public Hero(int x, int y) {
        super(x, y);
    }

    //射击
    public void shotEnemyTank() {

        //发多颗子弹怎么办,控制在我们的面板上,最多只有5颗
        if (shots.size()==5){
            return;
        }
        switch (getDirect()) {
            case 0://上
                shot = new Shot(getX() + 20, getY(), 0);
                break;
            case 1://右边
                shot = new Shot(getX() + 50, getY() + 30, 1);
                break;
            case 2:
                shot = new Shot(getX() + 20, getY() + 60, 2);
                break;
            case 3:
                shot = new Shot(getX() - 10, getY() + 30, 3);
                break;
        }
        //把新创建的shot放入到shots
        shots.add(shot);
        //启动我们的Shot线程
        new Thread(shot).start();
    }
}

EnemyTank

package tankgame05;

import java.util.Vector;

//敌方坦克
public class EnemyTank extends Tank implements Runnable {
    //在敌人坦克类,使用Vector保存多个Shot
    Vector<Shot> shots = new Vector<>();


    public EnemyTank(int x, int y) {
        super(x, y);
    }

    @Override
    public void run() {
        while (true) {

            //这里我们判断如果shots size() =0,创建一颗子弹,放入到
            //shots集合,并启动
            if (isLive && shots.size() <= 5) {//所以敌人同时只能存在五个子弹
                Shot s = null;
                switch (getDirect()) {
                    case 0://上
                        s = new Shot(getX() + 20, getY(), 0);
                        break;
                    case 1://右边
                        s = new Shot(getX() + 50, getY() + 30, 1);
                        break;
                    case 2:
                        s = new Shot(getX() + 20, getY() + 60, 2);
                        break;
                    case 3:
                        s = new Shot(getX() - 10, getY() + 30, 3);
                        break;
                }
                shots.add(s);
                new Thread(s).start();
            }

            //根据坦克方向继续移动
            for (int i = 0; i < 30; i++) {
                switch (getDirect()) {
                    case 0://向上
                        //让坦克保持一个方向,走30步
                        moveUp();
                        break;
                    case 1://向右
                        moveRight();
                        break;
                    case 2://向下
                        moveDown();
                        break;
                    case 3://向左
                        moveLeft();
                        break;
                }
                //休眠50毫秒
                try {
                    Thread.sleep(50);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            //然后随机的改变坦克方向 0-3
            setDirect((int) (Math.random() * 4));
            if (!isLive) {
                break;
            }
        }
    }
}

Shot

package tankgame05;
//射击事件
public class Shot implements Runnable {
    private int x;
    private int y;
    private int direct;//方向 0上 1右 2下 3左
    private int speed = 10;//子弹速度
    boolean isAlive = true;

    public Shot(int x, int y, int direct) {
        this.x = x;
        this.y = y;
        this.direct = direct;
    }

    public int getX() {
        return x;
    }

    public int getY() {
        return y;
    }

    @Override
    public void run() {
        while (true) {
            //休眠
            try {
                Thread.sleep(100);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            //根据方向移动
            switch (direct) {
                case 0:
                    y -= speed;
                    break;
                case 1:
                    x += speed;
                    break;
                case 2:
                    y += speed;
                    break;
                case 3:
                    x -= speed;
                    break;
            }
//            System.out.println("子弹坐标(" + x + "," + y + ")");
            //当子弹移动到面板的边界时,就应该销毁(把启动的子弹的线程销毁)
            // 当子弹碰到敌人坦克时,也应该结束线程
            if (x < 0 || x >= 1000 || y < 0 || y >= 750 || isAlive == false) {
//                System.out.println("子弹线程退出");
                isAlive = false;
                break;
            }
        }
    }
}

Bomb

package tankgame05;
//爆炸事件

public class Bomb {
    int x, y;//炸弹的坐标
    int life = 9;//炸弹的生命周期
    boolean isLive = true;//是否还存活

    public Bomb(int x, int y) {
        this.x = x;
        this.y = y;
    }

    //减少爆炸效果的生命值
    public void lifeDown() {//配合出现图片的爆炸效果
        if (life > 0) {
            life--;
        } else {
            isLive = false;
        }
    }
}

MyPanel

package tankgame05;
//面板

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

////为了监听键盘事件,实现KeyListener
//为了让Panel不停的重绘子弹,需要将MyPanel实现Runnable ,当做一个线程使用
public class MyPanel extends JPanel implements KeyListener, Runnable {
    Hero hero = null;//定义我的坦克
    Vector<EnemyTank> enemyTanks = new Vector<>();//定义敌人坦克
    //定义一个Vector ,用于存放多个爆炸效果
    //说明,当子弹击中坦克时,加入一个Bomb对象到bombs
    Vector<Bomb> bombs = new Vector<>();
    int enemyTankSize = 3;

    //定义三张炸弹图片,用于显示爆炸效果
    Image image1 = null;
    Image image2 = null;
    Image image3 = null;

    public MyPanel() {
        hero = new Hero(800, 100);//初始化自己的坦克位置
        for (int i = 0; i < enemyTankSize; i++) {//初始化敌人的坦克位置
            EnemyTank enemyTank = new EnemyTank(100 * (i + 1), 0);
            enemyTank.setDirect(2);//设置坦克方向
            //启动敌人坦克线程,让他们动起来
            new Thread(enemyTank).start();

//            Shot shot = new Shot(enemyTank.getX() + 20, enemyTank.getY() + 60, 2);//创建敌人子弹
//            enemyTank.shots.add(shot);//加入enemyTanks的Vector集合
//            //启动shot对象
//            new Thread(shot).start();//敌方子弹移动

            enemyTanks.add(enemyTank);//加入敌方坦克集合
        }
//        hero.setSpeed(5);
        image1 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(1).png"));
        image2 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(2).png"));
        image3 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/(3).png"));

        //图片在被第一次使用时才真正加载
        // 所以会出现第一次爆炸的图程序找不到还没加载好图片导致这次爆炸效果缺失的情况。
        //因此要提前自定义爆炸一次
//        bombs.add(new Bomb(-100,-100));
    }

    @Override
    public void paint(Graphics g) {
        super.paint(g);

        g.fillRect(0, 0, 1000, 750);//黑色背景
        if (hero!=null&&hero.isLive){//画出hero坦克
            drawTank(hero.getX(), hero.getY(), g, hero.getDirect(), 0);
        }

        //画出hero发出的子弹
//        if (hero.shot != null && hero.shot.isAlive == true) {
//            g.fill3DRect(hero.shot.getX() - 1, hero.shot.getY() - 1, 3, 3, false);
//        }
        //将hero的子弹集合shots ,遍历取出绘制
        for (int i = 0; i < hero.shots.size(); i++) {
            Shot shot = hero.shots.get(i);
            if (shot != null && shot.isAlive) {
                g.fill3DRect(shot.getX() - 1, shot.getY() - 1, 3, 3, false);
            }else {//如果该shot对象已经无效,就从shots集合中拿掉
                hero.shots.remove(shot);
            }
        }

        for (int i = 0; i < bombs.size(); i++) {
            //取出炸弹
            Bomb bomb = bombs.get(i);
            //根据当前这个bomb对象的life值去画出对应的图片
            if (bomb.life > 6) {
                g.drawImage(image1, bomb.x, bomb.y, 60, 60, this);
            } else if (bomb.life > 3) {
                g.drawImage(image2, bomb.x, bomb.y, 60, 60, this);
            } else {
                g.drawImage(image3, bomb.x, bomb.y, 60, 60, this);
            }
            //让这个炸弹的生命值减少
            bomb.lifeDown();
            //如果bomb life为0,就从bombs 的集合中删除
            if (bomb.life <= 0) {
                bombs.remove(bomb);
            }
        }

        //画出敌人坦克
        for (int i = 0; i < enemyTanks.size(); i++) {
            //取出坦克
            EnemyTank enemyTank = enemyTanks.get(i);
            if (enemyTank.isLive) {//判断敌方坦克是否存活
                drawTank(enemyTank.getX(), enemyTank.getY(), g, enemyTank.getDirect(), 1);
                //画出 enemyTank所有子弹
                for (int j = 0; j < enemyTank.shots.size(); j++) {
                    //取出子弹
                    Shot shot = enemyTank.shots.get(j);//绘制
                    if (shot.isAlive) { //isAlive = true
                        g.draw3DRect(shot.getX() - 1, shot.getY() - 1, 3, 3, false);
                    } else {
                        //从Vector移除
                        enemyTank.shots.remove(shot);
                    }
                }
            }
        }

    }

    /**
     * @param x      坦克左上角x坐标
     * @param y      坦克左上角y坐标
     * @param g      画笔
     * @param direct 坦克方向(上下左右)
     * @param type   坦克类型
     */
    public void drawTank(int x, int y, Graphics g, int direct, int type) {

        switch (type) {
            case 0://我们的坦克
                g.setColor(Color.cyan);
                break;
            case 1://敌人的坦克
                g.setColor(Color.yellow);
                break;
        }
        //根据坦克方向,绘制坦克
        switch (direct) {
            case 0://向上
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y);//画出炮筒
                break;
            case 1://右
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 50, y + 30);//画出炮筒
                break;
            case 2://下
                g.fill3DRect(x, y, 10, 60, false);//画出坦克左边轮子
                g.fill3DRect(x + 30, y, 10, 60, false);//画出坦克右边轮子
                g.fill3DRect(x + 10, y + 10, 20, 40, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x + 20, y + 60);//画出炮筒
                break;
            case 3://左
                g.fill3DRect(x - 10, y + 10, 60, 10, false);//画出坦克左边轮子
                g.fill3DRect(x - 10, y + 40, 60, 10, false);//画出坦克右边轮子
                g.fill3DRect(x, y + 20, 40, 20, false);//画出坦克盖子
                g.fillOval(x + 10, y + 20, 20, 20);//画出圆形盖子
                g.drawLine(x + 20, y + 30, x - 10, y + 30);//画出炮筒
                break;
        }

    }

    //如果我们的坦克可以发射多个子弹
    //在判断我方子弹是否击中敌人坦克时,就需要把我们的子弹集合中
    //所有的子弹,都取出和敌人的所有坦克,进行判断
    public void hitEnemyTank(){
        //遍历我们的子弹
        for (int j = 0; j < hero.shots.size(); j++) {
            Shot shot = hero.shots.get(j);
            if (shot != null && shot.isAlive) {
                //遍历敌人所有坦克,实时判断坦克是否被击中
                for (int i = 0; i < enemyTanks.size(); i++) {
                    EnemyTank enemyTank = enemyTanks.get(i);
                    hitTank(shot, enemyTank);
                }
            }
        }
    }

    //编写方法,判断我方的子弹是否击中敌人坦克
    //什么时候判断我方的子弹是否击中敌人坦克 ?run方法
    public void hitTank(Shot s, Tank enemyTank) {
        switch (enemyTank.getDirect()) {
            case 0://上
            case 2://下
                if (s.getX() > enemyTank.getX() && s.getX() < enemyTank.getX() + 40 &&
                        s.getY() > enemyTank.getY() && s.getY() < enemyTank.getY() + 60) {
                    s.isAlive = false;//设置子弹为死亡状态
                    enemyTank.isLive = false;//设置该坦克为死亡状态
                    //创建Bomb对象,加入到bombs集合
                    Bomb bomb = new Bomb(enemyTank.getX() - 10, enemyTank.getY());
                    bombs.add(bomb);
                    //当我的子弹击中敌人坦克后,将enemyTank 从lector拿掉
                    enemyTanks.remove(enemyTank);//移除该坦克
                }
                break;
            case 1:
            case 3:
                if (s.getX() > enemyTank.getX() - 10 && s.getX() < enemyTank.getX() + 50 &&
                        s.getY() > enemyTank.getY() + 10 && s.getY() < enemyTank.getY() + 50) {
                    s.isAlive = false;
                    enemyTank.isLive = false;
                }
                break;
        }
    }
    //判断敌方的子弹是否击中我方坦克
    public void hitHeroTank(){
        //遍历敌方子弹
        for (int i = 0; i < enemyTanks.size(); i++) {
            EnemyTank enemyTank = enemyTanks.get(i);
            for (int j = 0; j < enemyTank.shots.size(); j++) {
                Shot shot = enemyTank.shots.get(j);
                hitTank(shot,hero);
            }
        }
    }


    @Override
    public void keyTyped(KeyEvent e) {
    }

    //处理WDSA键按下情况
    @Override
    public void keyPressed(KeyEvent e) {
        if (e.getKeyCode() == KeyEvent.VK_W) {
            //改变坦克方向
            hero.setDirect(0);
            //修改坦克坐标
            hero.moveUp();
        } else if (e.getKeyCode() == KeyEvent.VK_D) {
            hero.setDirect(1);
            hero.moveRight();
        } else if (e.getKeyCode() == KeyEvent.VK_S) {
            hero.setDirect(2);
            hero.moveDown();
        } else if (e.getKeyCode() == KeyEvent.VK_A) {
            hero.setDirect(3);
            hero.moveLeft();
        }

        if (e.getKeyCode() == KeyEvent.VK_J) {
            //判断hero的子弹是否销毁,发射一颗子弹
//            if (hero.shot == null || !hero.shot.isAlive) {
//                hero.shotEnemyTank();
//            }
            //发射多颗子弹
            hero.shotEnemyTank();
        }
//        this.repaint();
    }

    @Override
    public void keyReleased(KeyEvent e) {

    }

    @Override
    public void run() {
        while (true) {
            try {
                Thread.sleep(100);//每停0.1s执行一次
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            hitEnemyTank();//检测子弹是否打中坦克
            hitHeroTank();
            this.repaint();//刷新面板

        }
    }
}

HspTankGame05

package tankgame05;
//窗体

import javax.swing.*;

public class HspTankGame05 extends JFrame {
    MyPanel mp=null;
    public static void main(String[] args) {
        new HspTankGame05();
    }

    public HspTankGame05() {
        mp=new MyPanel();
        new Thread(mp).start();//另外开一线程一直刷新面板内容和面板中一些需要同时执行的程序
        this.add(mp);
        this.setSize(1016,789);
        this.addKeyListener(mp);
        this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        this.setVisible(true);
    }
}
posted @ 2022-09-02 19:51  文杰2000  阅读(90)  评论(0编辑  收藏  举报