项目-坦克大战

devtools/2024/9/23 4:23:02/

增加功能

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

  • 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/devtools/42106.html

相关文章

list使用

C的list是C STL&#xff08;标准模板库&#xff09;中的容器&#xff0c;它是一个双向链表&#xff0c;允许在容器的任意位置进行快速插入和删除操作。list的特点是它能够以O(1)的时间复杂度在容器的任意位置插入和删除节点。 使用list需要包含头文件&#xff0c;并且使用std命…

【寒枫顾辞老航小说传】第一回:梦回大唐

在秋日的黄昏&#xff0c;枫叶如烈火般燃烧&#xff0c;寒枫、老航和阿辞&#xff0c;三位好友坐在公园的长椅上&#xff0c;谈笑风生。他们相识多年&#xff0c;性格迥异&#xff0c;却情同手足。寒枫温文尔雅&#xff0c;老航机智幽默&#xff0c;阿辞则豪放不羁。这一天&…

C语言经典例题-18

1.判断是不是字母 题目描述: KK想判断输入的字符是不是字母&#xff0c;请帮他编程实现。 输入描述: 多组输入&#xff0c;每一行输入一个字符。 输出描述: 针对每组输入&#xff0c;输出单独占一行&#xff0c;判断输入字符是否为字母&#xff0c;输出内容详见输出样例。 输…

写一个类ChatGPT应用,前后端数据交互有哪几种

❝ 对世界的态度&#xff0c;本质都是对自己的态度 ❞ 大家好&#xff0c;我是「柒八九」。一个「专注于前端开发技术/Rust及AI应用知识分享」的Coder 前言 最近&#xff0c;公司有一个AI项目&#xff0c;要做一个文档问答的AI产品。前端部分呢&#xff0c;还是「友好借鉴」Cha…

Helm安装集群整理

这里写目录标题 1.添加nfs存储2.安装redis集群3.安装neo4j集群4.安装clickhouse集群5. 安装zookeeper集群6. 安装es集群7. 安装openebs8.安装radondb 1.添加nfs存储 项目地址&#xff1a;https://github.com/kubernetes-sigs/nfs-subdir-external-provisioner helm install nfs…

Golang RPC实现-day02

导航 Golang RPC实现一、客户端异步并发多个请求1、 客户端结构体2、 一个客户端&#xff0c;异步发送多个请求&#xff0c;使用call结构体代表客户端的每次请求3、客户端并发多个请求4、客户端接收请求 Golang RPC实现 day01 我们实现了简单的服务端和客户端。我们简单总结一…

【专用】C# ArrayList的用法总结

System.Collections.ArrayList类是一个特殊的数组。通过添加和删除元素&#xff0c;就可以动态改变数组的长度。 一、优点 1. 支持自动改变大小的功能 2. 可以灵活的插入元素 3. 可以灵活的删除元素 4. 可以灵活访问元素 二、局限性 跟一般的数组比起来&#xff0c;速度…

软考--试题六--享元模式(Flyweight)

享元模式(Flyweight) 意图 运用共享技术有效地支持大量细粒度的对象(将对象进行细分) 结构 适用性 1、一个应用程序使用了大量的对象 2、完全由于使用大量的对象&#xff0c;造成很大的存储开销 3、对象的大多数状态都快变为外部状态 4、如果删除对象的外部状态(易变)&…