项目-坦克大战

embedded/2024/10/21 7:31:35/

增加功能

我方坦克在发射的子弹消亡后,才能发射新的子弹。同时实现发多颗子弹

  • 1,在按下J键,我们判断当前hero对象的子弹,是否已经销毁
  • 2,如果没有销毁,就不去触发shotEnemyTank
  • 3,如果已经销毁,才去触发shotEnemyTank
  • 4,如果要发射多颗子弹,就使用Vector保存(hero类)
  • 5,在绘制我方子弹时,需要遍历Vector,在按键J时,再次调用shotEnemyTank
  • 6,如果需要控制在我们的面板上最多只有5颗子弹:用if语句进行判断即可(hero类),

Hero类

java">import java.util.Vector;//自己的坦克,继承坦克父类
public class Hero extends Tank {//定义一个Shot对象,表示一个射击(线程)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;}//创建Shot对象,根据当前Hero对象的位置和方向来创建Shotswitch (getDirect()){//得到Hero对象方向case 0://上shot = new Shot(getX()+20,getY(),0);break;case 1://右shot = new Shot(getX()+60,getY()+20,1);break;case 2://下shot = new Shot(getX()+20,getY()+60,2);break;case 3://左shot = new Shot(getX(),getY()+20,3);break;}//把新创建的shot放入到shots中shots.add(shot);//启动我们的shot线程new Thread(shot).start();}
}

画板类
改变keyPressed方法

java"> //当某个键按下,该方法会触发@Overridepublic void keyPressed(KeyEvent e) {if (e.getKeyCode() == KeyEvent.VK_W) {//上//改变坦克方向hero.setDirect(0);//让坦克动起来if (hero.getY()>0){hero.moveUp();}} else if (e.getKeyCode() == KeyEvent.VK_S) {//下hero.setDirect(2);if (hero.getY()+60<750){hero.moveDown();}} else if (e.getKeyCode() == KeyEvent.VK_A) {//左hero.setDirect(3);if (hero.getX()>0){hero.moveLeft();}} else if (e.getKeyCode() == KeyEvent.VK_D) {//右hero.setDirect(1);if (hero.getX()+60<1000)hero.moveRight();}//如果用户按下的是J,就发射if (e.getKeyCode() == KeyEvent.VK_J) {//判断hero的子弹是否销毁---发射一颗子弹if (hero.shot==null|| !hero.shot.isLive) {//线程销毁后,不代表shot对象为空,所以还需要判断shot的声明周期,才可以继续发子弹hero.shotEnemyTank();}//发射多颗子弹hero.shotEnemyTank();}//重绘repaint();}

在paint方法中遍历Vector集合

java">  @Overridepublic void paint(Graphics g) {super.paint(g);//3,通过画笔填充一个矩形区域   g.fillRect();g.fillRect(0, 0, 1000, 750);//填充矩形,默认为黑色//画出自己的坦克drawTank(hero.getX(), hero.getY(), g, hero.getDirect(), 1);
//        drawTank(hero.getX() +100,hero.getY(),g,hero.getDirect(),0);
//        drawTank(hero.getX() +200,hero.getY(),g,hero.getDirect(),0);
//        drawTank(hero.getX() +300,hero.getY(),g,hero.getDirect(),1);//判断何时画子弹
//        if (hero.shot != null && hero.shot.isLive == true) {
//            g.setColor(Color.white);
//            g.draw3DRect(hero.shot.x, hero.shot.y, 1, 1, false);
//
//        }//将hero的子弹集合shots,遍历取出绘制for (int i = 0; i < hero.shots.size(); i++) {Shot shot = hero.shots.get(i);if (shot != null && shot.isLive == true) {g.setColor(Color.white);g.draw3DRect(shot.x, shot.y, 1, 1, false);}else {//如果该shot对象已经无效,就从shots集合中拿掉hero.shots.remove(shot);}}//如果bombs 集合中有对象,就画出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++) {//从Vector取出坦克EnemyTank enemyTank = enemyTanks.get(i);//判断当前坦克是否还存活if (enemyTank.isLive) {//当敌人坦克时存活的,才画出该坦克drawTank(enemyTank.getX(), enemyTank.getY(), g, enemyTank.getDirect(), 0);//画出 enemyTank 所有子弹for (int j = 0; j < enemyTank.shots.size(); j++) {//取出子弹Shot shot = enemyTank.shots.get(j);//绘制if (shot.isLive) {//isLive=trueg.setColor(Color.white);g.draw3DRect(shot.x, shot.y, 1, 1, false);} else {//从Vector 移除enemyTank.shots.remove(shot);}}}}}

让敌人坦克发射的子弹消亡后,可以再发射子弹

  • 1,在敌人坦克类中的run方法中判断 Vector集合的数量
  • 2,如果仍然存活且数量小于1,则执行相关业务代码
java">//敌人坦克
public class EnemyTank extends Tank implements Runnable {//给敌人坦克类,使用Vector 保存多个ShotVector<Shot> shots = new Vector<>();//定义敌人坦克的存货属性boolean isLive = true;public EnemyTank(int x, int y) {super(x, y);}@Overridepublic void run() {while (true) {//这里我们判断如果shots size() = 0; 创建一颗子弹,放入到shots集合,并启动if (isLive&&shots.size()<1){Shot s = null;//判断坦克的方向,创建对应的子弹switch (getDirect()){case 0:s=new Shot(getX()+20,getY(),0);case 1:s=new Shot(getX()+60,getY()+20,0);case 2:s=new Shot(getX()+20,getY()+60,0);case 3:s=new Shot(getX(),getY()+20,0);}shots.add(s);new Thread(s).start();}switch (getDirect()) {//根据坦克的方向来进行移动case 0://向上//让坦克保持一个方向,走30步for (int i = 0; i < 30; i++) {if (getY()>0){moveUp();}//休眠50好眠try {Thread.sleep(50);} catch (InterruptedException e) {throw new RuntimeException(e);}}break;case 1://向右//让坦克保持一个方向,走30步for (int i = 0; i < 30; i++) {if (getX()+60<1000){moveRight();}//休眠50好眠try {Thread.sleep(50);} catch (InterruptedException e) {throw new RuntimeException(e);}}break;case 2://向下//让坦克保持一个方向,走30步for (int i = 0; i < 30; i++) {if (getY()+60<750){moveDown();}//休眠50好眠try {Thread.sleep(50);} catch (InterruptedException e) {throw new RuntimeException(e);}}break;case 3://向左//让坦克保持一个方向,走30步for (int i = 0; i < 10; i++) {if (getX()>0){moveLeft();}//休眠50好眠try {Thread.sleep(50);} catch (InterruptedException e) {throw new RuntimeException(e);}}break;}//休眠50好眠try {Thread.sleep(50);} catch (InterruptedException e) {throw new RuntimeException(e);}// 然后随机的改变坦克方向0-3setDirect((int) (Math.random() * 4));  //0-3//一旦写并发程序,一定要考虑清楚,该线程什么时候结束if (!isLive) {break;}}}
}

当敌人的坦克击中我方坦克时,我方坦克消失,并出现爆炸效果

  • 1,编写方法,判断敌人坦克是否击中我方坦克
  • 2,在run方法中调用该方法
java">//画板类
public class MyPanel extends JPanel implements KeyListener, Runnable {//定义我的坦克Hero hero = null;//定义敌人坦克,放入到Vector(线程安全,可以有多个)Vector<EnemyTank> enemyTanks = new Vector<>();int enemyTankSize = 3;//定义一个Vector,用于存放炸弹(炸弹既不属于我方坦克也不属于地方坦克 ,所以坦克放在画板类)//当我们的子弹击中坦克时,就加入一个Bomb对象到bombsVector<Bomb> bombs = new Vector<>();//定义三张炸弹图片,用于显示爆炸效果Image image1 = null;Image image2 = null;Image image3 = null;public MyPanel() {//初始化自己的坦克hero = new Hero(100, 100);hero.setSpeed(5);//初始化敌人的坦克for (int i = 0; i < enemyTankSize; i++) {//创建一个敌人的坦克EnemyTank enemyTank = new EnemyTank(100 * (i + 1), 0);//启动敌人坦克线程,让他动起来new Thread(enemyTank).start();//设置方向enemyTank.setDirect(2);//给该enemyTank 加入一颗子弹Shot shot = new Shot(enemyTank.getX() + 20, enemyTank.getY() + 60, enemyTank.getDirect());enemyTank.shots.add(shot);//启动shot对象new Thread(shot).start();//把敌人的坦克加入的总的坦克对象中enemyTanks.add(enemyTank);}
//        image1 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/1.gif"));
//        image2 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/2.gif"));
//        image3 = Toolkit.getDefaultToolkit().getImage(Panel.class.getResource("/3.gif"));//初始化照片对象image1 = new ImageIcon("image/1.gif").getImage();image2 = new ImageIcon("image/2.gif").getImage();image3 = new ImageIcon("image/3.gif").getImage();}@Overridepublic void paint(Graphics g) {super.paint(g);//3,通过画笔填充一个矩形区域   g.fillRect();g.fillRect(0, 0, 1000, 750);//填充矩形,默认为黑色if (hero != null && hero.isLive){//画出自己的坦克drawTank(hero.getX(), hero.getY(), g, hero.getDirect(), 1);}
//        drawTank(hero.getX() +100,hero.getY(),g,hero.getDirect(),0);
//        drawTank(hero.getX() +200,hero.getY(),g,hero.getDirect(),0);
//        drawTank(hero.getX() +300,hero.getY(),g,hero.getDirect(),1);//判断何时画子弹
//        if (hero.shot != null && hero.shot.isLive == true) {
//            g.setColor(Color.white);
//            g.draw3DRect(hero.shot.x, hero.shot.y, 1, 1, false);
//
//        }//将hero的子弹集合shots,遍历取出绘制for (int i = 0; i < hero.shots.size(); i++) {Shot shot = hero.shots.get(i);if (shot != null && shot.isLive == true) {g.setColor(Color.white);g.draw3DRect(shot.x, shot.y, 1, 1, false);}else {//如果该shot对象已经无效,就从shots集合中拿掉hero.shots.remove(shot);}}//如果bombs 集合中有对象,就画出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++) {//从Vector取出坦克EnemyTank enemyTank = enemyTanks.get(i);//判断当前坦克是否还存活if (enemyTank.isLive) {//当敌人坦克时存活的,才画出该坦克drawTank(enemyTank.getX(), enemyTank.getY(), g, enemyTank.getDirect(), 0);//画出 enemyTank 所有子弹for (int j = 0; j < enemyTank.shots.size(); j++) {//取出子弹Shot shot = enemyTank.shots.get(j);//绘制if (shot.isLive) {//isLive=trueg.setColor(Color.white);g.draw3DRect(shot.x, shot.y, 1, 1, false);} else {//从Vector 移除enemyTank.shots.remove(shot);}}}}}/*int x 坦克的横坐标int y 坦克的纵坐标Graphics g  画笔int direct  坦克的方向int 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;}//根据坦克的方向,来绘制坦克//direct表示方法:  0:向上   1:向右    2:向下    3:向左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, y, 60, 10, false);//画出坦克上边轮子g.fill3DRect(x, y + 30, 60, 10, false);//画出坦克下边轮子g.fill3DRect(x + 10, y + 10, 40, 20, false);//画出坦克盖子g.fillOval(x + 20, y + 10, 20, 20);//画出圆形盖子g.drawLine(x + 30, y + 20, x + 60, y + 20);//画出炮筒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, y, 60, 10, false);//画出坦克上边轮子g.fill3DRect(x, y + 30, 60, 10, false);//画出坦克下边轮子g.fill3DRect(x + 10, y + 10, 40, 20, false);//画出坦克盖子g.fillOval(x + 20, y + 10, 20, 20);//画出圆形盖子g.drawLine(x + 30, y + 20, x, y + 20);//画出炮筒break;default:System.out.println("暂时没有处理");}}//编写方法,判断敌人坦克是否击中我方坦克public void hitHero(){//遍历所有的敌人坦克for (int i = 0; i < enemyTanks.size(); i++) {//取出敌人坦克EnemyTank enemyTank = enemyTanks.get(i);//遍历enemyTank 对象的所有子弹for (int j = 0; j < enemyTank.shots.size(); j++) {//取出子弹Shot shot = enemyTank.shots.get(j);//判断shot 是否击中我的坦克if (hero.isLive&&shot.isLive){hitTank( shot,hero);}}}}public void hitEnemy(){//判断我们的子弹是否击中了敌人坦克if (hero.shot != null && hero.shot.isLive) {//当我方的子弹还存活//遍历敌人所有的坦克for (int i = 0; i < enemyTanks.size(); i++) {EnemyTank enemyTank = enemyTanks.get(i);hitTank(hero.shot, enemyTank);}}}//编写方法:判断我方子弹是否击敌人坦克//什么时候判断,我方坦克是否击中敌人坦克?  run方法public void hitTank(Shot s, Tank tank) {//判断s 击中坦克switch (tank.getDirect()) {case 0://敌人坦克向上case 2://敌人坦克向下if (s.x > tank.getX() && s.y < tank.getY() + 40 &&s.y > tank.getY() && s.y < tank.getY() + 60) {s.isLive = false;tank.isLive = false;//当我的子弹击中敌人坦克后,将enenmyTank 从Vector 拿掉enemyTanks.remove(tank);//这里敌人坦克被击中//创建Bomb对象,加入到bombs集合Bomb bomb = new Bomb(tank.getX(), tank.getY());bombs.add(bomb);}break;case 1://敌人坦克向右case 3://敌人坦克向下if (s.x > tank.getX() && s.y < tank.getY() + 60 &&s.y > tank.getY() && s.y < tank.getY() + 40) {s.isLive = false;tank.isLive = false;Bomb bomb = new Bomb(tank.getX(), tank.getY());bombs.add(bomb);}break;}}//有字符输出时,该方法就会触发@Overridepublic void keyTyped(KeyEvent e) {}//当某个键按下,该方法会触发@Overridepublic void keyPressed(KeyEvent e) {if (e.getKeyCode() == KeyEvent.VK_W) {//上//改变坦克方向hero.setDirect(0);//让坦克动起来if (hero.getY()>0){hero.moveUp();}} else if (e.getKeyCode() == KeyEvent.VK_S) {//下hero.setDirect(2);if (hero.getY()+60<750){hero.moveDown();}} else if (e.getKeyCode() == KeyEvent.VK_A) {//左hero.setDirect(3);if (hero.getX()>0){hero.moveLeft();}} else if (e.getKeyCode() == KeyEvent.VK_D) {//右hero.setDirect(1);if (hero.getX()+60<1000)hero.moveRight();}//如果用户按下的是J,就发射if (e.getKeyCode() == KeyEvent.VK_J) {//判断hero的子弹是否销毁---发射一颗子弹if (hero.shot==null|| !hero.shot.isLive) {//线程销毁后,不代表shot对象为空,所以还需要判断shot的声明周期,才可以继续发子弹hero.shotEnemyTank();}//发射多颗子弹hero.shotEnemyTank();}//重绘repaint();}//当某个键释放(松开),该方法会触发@Overridepublic void keyReleased(KeyEvent e) {}//添加线程的方法,确保子弹可以重绘@Overridepublic void run() {//每隔 100毫秒,重绘区域,刷新绘图区域,子弹就移动while (true) {try {Thread.sleep(100);} catch (InterruptedException e) {throw new RuntimeException(e);}//判断我们坦克是否击中别人hitEnemy();//判断敌人坦克是否击中我们hitHero();this.repaint();}}
}

http://www.ppmy.cn/embedded/41936.html

相关文章

Microsoft 安全Copilot:适时而生的得力工具

随着数字化转型的加速&#xff0c;网络安全威胁日益增多&#xff0c;Microsoft Copilot for Security的出现恰逢其时&#xff0c;它将帮助我们更好地应对这些挑战&#xff0c;保护我们的数据和系统免受攻击。 ✦什么是Microsoft 安全Copilot✦ 概述 Microsoft 安全 Copilot 是…

SAP BSEG VS ACDOCA 差异

温习一下 ACDOCA VS BSEG matinal&#xff1a;S4 HANA 详解ACDOCA与BSEG的区别都在这了_sap acdoca-CSDN博客

java如何处理大文件

Java中处理大文件时&#xff0c;通常需要采取一些特定的策略来避免内存溢出或性能问题。以下是一些处理大文件的建议&#xff1a; 使用流&#xff08;Streams&#xff09;&#xff1a; 使用InputStream和OutputStream的派生类&#xff08;如FileInputStream, BufferedInputStr…

【RAG 论文】IRCoT:基于 CoT 的交叉检索解决多步骤问题

论文&#xff1a;Interleaving Retrieval with Chain-of-Thought Reasoning for Knowledge-Intensive Multi-Step Questions ⭐⭐⭐⭐ ACL 2023, arXiv:2212.10509 Code: github.com/stonybrooknlp/ircot 论文速读 大多数 RAG 都是一次检索来辅助 LLM 生成&#xff0c;但是面对…

springboot2.x集成Elasticsearch7.7.0

一、前言 elasticsearch安装就不做过多介绍了&#xff0c;网上一搜一大堆&#xff1b;最需要注意的就是Elasticsearch与spring版本&#xff0c;防止版本不兼容导致的后续的一系列问题。我这里springbootspring-data-elasticsearch&#xff0c;他们的版本对照关系可以参照sprin…

windows快速计算文件的SHA256数值的步骤

在文件路径打开cmd窗口 输入命令 用Windows自带的certutil命令来计算一个文件的校验值1&#xff1a; certutil支持的算法有&#xff1a;MD2 MD4 MD5 SHA1 SHA256 SHA384 SHA512。 certutil的使用方法非常简单&#xff0c;只需要执行“certutil -hashfile 文件名 校验值类型”…

【35分钟掌握金融风控策略28】贷中模型体系策略应用

目录 贷中模型体系策略应用 信用模型体系和模型在策略中的应用 反欺诈模型体系和模型在策略中的应用 运营模型体系和模型在策略中的应用 贷中模型体系策略应用 在贷前模型部分已经讲过&#xff0c;贷前开发的很多模型是可以在贷中直接使用的。贷中与贷前的不同点在于&…

Java设计模式——装饰者模式

Java设计模式——装饰者模式 1.简介 装饰者模式&#xff08;Decorator Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许用户通过在一个对象上动态地添加职责或行为来增强其功能。这种类型的设计模式属于对象结构型模式&#xff0c;在不需要改变一个对象的内部结构…