13.4-全栈Java笔记:打飞机游戏实战项目|offScreenImage|GameObject|Plane

双缓冲技术解决闪烁问题

上节,我们实现了动画效果,但是发现窗口会不停的闪烁,体验度非常差。在实际开发中,绘制图形是非常复杂的,绘图可能需要几秒甚至更长时间,也经常发生闪烁现象, 为了解决这个问题,我们通常使用“双缓冲技术”。

1)“双缓冲技术”的绘图过程如下:

a. 在内存中创建与画布一致的缓冲区

b. 在缓冲区画图

c. 将缓冲区位图拷贝到当前画布上

e. 释放内存缓冲区

双缓冲即在内存中创建一个与屏幕绘图区域一致的对象,先将图形绘制到内存中的这个对象上,再一次性将这个对象上的图形拷贝到屏幕上,这样能大大加快绘图的速度。

我们只需将如下“双缓冲”实现代码,放入MyGrameFrame类中,即可:

【示例1】添加双缓冲技术


Image offScreenImage = null;

public void update(Graphics g) {

if(offScreenImage == null)

offScreenImage = this.createImage(500,500);//这是游戏窗口的宽度和高度

Graphics gOff = offScreenImage.getGraphics();

paint(gOff);

g.drawImage(offScreenImage, 0, 0, null);

}

GameObject类设计

1) GameObject类的定义

我们发现,窗口中所有的对象(飞机、炮弹等等)都有很多共性:“图片对象、坐标位置、运行速度、宽度和高度”。为了方便程序开发,我们需要设计一个GameObject类,它可以作为所有游戏物体的父类,方便我们编程。

【示例2】GameObject类


package cn.sxt.game;

import java.awt.Graphics;

import java.awt.Image;

import java.awt.Rectangle;

public class GameObject {

Image img;        //该物体对应的图片对象

double x,y;       //该物体的坐标

int speed;          //该物体的运行速度

int width,height;     //该物体所在矩形区域的宽度和高度

/**

* 怎么样绘制本对象

* @param g

*/

public void drawMySelf(Graphics  g){

g.drawImage(img, (int)x, (int)y, null);

}

public GameObject(Image img, double x, double y) {

this.img = img;

this.x = x;

this.y = y;

if(img!=null){

this.width = img.getWidth(null);

this.height = img.getHeight(null);

}

}

public GameObject(Image img, double x, double y, int speed, int width,

int height) {

this.img = img;

this.x = x;

this.y = y;

this.speed = speed;

this.width = width;

this.height = height;

}

public GameObject() {

}

/**

* 返回物体对应矩形区域,便于后续在碰撞检测中使用

* @return

*/

public Rectangle getRect(){

return    new Rectangle((int)x,(int) y, width, height);

}

}

2) 设计飞机类

有了GameObject这个父类,我们设计飞机类特别简单,目前飞机类没有特别复杂的要求。我们只需简单的继承,即可使用:

【示例3】Plane类


package cn.sxt.game;

import java.awt.Graphics;

import java.awt.Image;

public class Plane  extends GameObject {

@Override

public void drawMySelf(Graphics g) {

super.drawMySelf(g);

this.x +=3;//飞机水平飞,我们也可以调整x、y算法,按照我们指定的路径飞行

}

public Plane(Image img, double x, double y) {

super(img,x,y);

}

}

通过继承,我们发现实现新的类,爽了很多!

3) MyGameFrame类调用方式的调整

我们将Plane类封装后,也无需在MyGameFrame类中添加那么多飞机的属性,我们全部封装到了Plane类里面,因此,调用也变得更加简单。

【示例4】封装后的MyGameFrame类


public class MyGameFrame extends Frame {

Image bgImg = GameUtil.getImage("images/bg.jpg");

Image planeImg = GameUtil.getImage("images/plane.png");

Plane plane = new Plane(planeImg,300,300);

//paint方法作用是:画出整个窗口及内部内容。被系统自动调用。

@Override

public void paint(Graphics g) {

g.drawImage(bgImg, 0, 0, null);

plane.drawMySelf(g);     //画出飞机本身

}

//其余代码,没有任何变化,不在附上,自行参考上一个版本。

}

通过面向对象封装后,如果我们要再创建多个飞机,也变得异常简单。

【示例5】创建多个飞机


public class MyGameFrame extends Frame {

Image bgImg = GameUtil.getImage("images/bg.jpg");

Image planeImg = GameUtil.getImage("images/plane.png");

Plane plane = new Plane(planeImg,300,300);

Plane plane2 = new Plane(planeImg,300,350);

Plane plane3 = new Plane(planeImg,300,400);

//paint方法作用是:画出整个窗口及内部内容。被系统自动调用。

@Override

public void paint(Graphics g) {

g.drawImage(bgImg, 0, 0, null);

plane.drawMySelf(g);     //画出飞机本身

plane2.drawMySelf(g);  //画出飞机本身

plane3.drawMySelf(g);  //画出飞机本身

}

//其余代码,和上个版本一致,为节省篇幅突出重点,不在附上。

}



「全栈Java笔记」是一部能帮大家从零到一成长为全栈Java工程师系列笔记。笔者江湖人称 Mr. G,10年Java研发经验,曾在神州数码、航天院某所研发中心从事软件设计及研发工作,从小白逐渐做到工程师、高级工程师、架构师。精通Java平台软件开发,精通JAVAEE,熟悉各种流行开发框架。

笔记包含从浅入深的六大部分:

A-Java入门阶段

B-数据库从入门到精通

C-手刃移动前端和Web前端

D-J2EE从了解到实战

E-Java高级框架精解

F-Linux和Hadoop

时间: 2024-08-27 08:31:16

13.4-全栈Java笔记:打飞机游戏实战项目|offScreenImage|GameObject|Plane的相关文章

13.8-全栈Java笔记:打飞机游戏实战项目|Explode|MyGameFrame|Plane

爆炸效果的实现 飞机被炮弹击中后,需要出现一个爆炸效果,让我们的画面更刺激. 爆炸效果的实现在游戏开发中也很常见. 我们定义Exlode类来表示爆炸的信息,爆炸类和普通类不一样的地方在于他实际上存储了一系列爆炸的图片,然后,进行轮播.最后,我们看到的就是一组酷炫的效果. 我们这里准备了一系列爆炸图片: 从爆炸开始的一个小火球到大火球,再到消失时的小火球.爆炸对象只需轮流加载这些图片即可. 我们将这些图片拷贝到项目下面,新建:images/explode文件夹,并将16张图片拷贝到文件夹下面. 爆

13.7-全栈Java笔记:打飞机游戏实战项目|Rectangle|intersects|Plane

碰撞类检测技术 游戏中,碰撞是遇到最频繁的技术.当然,很多游戏引擎内部已经做了碰撞检测处理,我们只需调用即可.本节课是从碰撞的原理进行讲解,大家自己去实现基本的碰撞检测. 矩形检测原理 游戏中,多个元素是否碰到一起,实际上,通常是用"矩形检测"原理实现的. 我们在前面提到,游戏中所有的物体都可以抽象成"矩形",我们只需判断两个矩形是否相交即可.对于一些复杂的多边形.不规则物体,实际上是将他分解成多个矩形,继续进行矩形检测. Java的API中,为我们提供了Recta

13.1-全栈Java笔记:打飞机游戏实战项目|AWT技术|MyGameFrame

简介和项目目标 通过游戏项目学习整个Java基础知识体系,我们做了精心的设计,让每一章知识都能获得应用. 比如:多线程用来实现动画效果.容器实现对于多发炮弹的存取和处理.常用类等等的应用. 寓教于乐,让大家迅速入门,更希望通过喜闻乐见的小游戏,让大家爱上编程,爱上"程序员". 老鸟建议 很多朋友会疑惑:"游戏项目,又不能拿到企业面试中,为什么要讲?" 这是一种太过于功利的想法.就像,我们说:"今天吃个馒头,又不是长高,为什么要吃呢?" 游戏项目的

13.3-全栈Java笔记:打飞机游戏实战项目|PaintThread|launchFrame

多线程和内部类实现动画效果 1)增加绘制窗口的线程类 前三个版本,我们步步为营,每个小版本都有功能的突破.但是,目前为止我们的窗口仍然是静态的,并没有像真正的游戏窗口那样"各种动.各种炫".本节我们结合多线程实现动画效果. 我们在MyGameFrame类中定义"重画窗口线程PaintThread类",为了方便使用MyGameFrame类的属性和方法,我们将PaintThread定义成内部类. [示例1]MyGameFrame类:增加PaintThread内部类 pu

13.6-全栈Java笔记:打飞机游戏实战项目|Shell|speed|launchFrame

炮弹类设计 通过炮弹类的设计,我们可以更深入了解构造器的用法以及容器的用法.同时,可能还需要读者稍微回忆一下初中数学曾学过的三角函数,这样更能理解炮弹飞行路径的计算原理. 当然,如果忘记这些知识了也没关系,毕竟在实际开发中很少涉及数学原理性的内容. 炮弹类基本设计 炮弹类我们用实心的黄色椭圆实现,不再加载新的图片.当然,大家课下可以自行找一些炮弹图片亦可. 我们的逻辑是在窗口固定位置(200,200)处生成炮弹,炮弹方向是随机的,并且遇到边界会反弹.  [示例1]Shell类 cn.sxt.ga

13.2-全栈Java笔记:打飞机游戏实战项目|Graphics|ImageIO|GameUtil

上节我们讲到如何在游戏项目中运用AWT技术绘制游戏窗口,本节我们继续往下讲~~  图形和文本绘制 1)paint方法 如果要在窗口中画图或者显示什么内容,我们需要重写paint(Graphics g)方法. 这个方法的作用是:画出整个窗口及其内部内容.它会被系统自动调用.我们自己不需要去调用这个方法. [示例1]paint方法介绍 @Override public void paint(Graphics g) { //paint方法作用是:画出整个窗口及内部内容.被系统自动调用. } 2)Gra

13.9-全栈Java笔记:打飞机游戏实战项目|Time|Plane|其他案例展示

其他功能 完成了基本的功能,这时候体验度还是很一般.为了让玩家更愿意玩我们的游戏,增加一些锦上添花的功能就很有必要.比如:游戏计时功能.全网排名等等.  计时功能 我们希望在玩游戏时,增加计时功能,可以清晰的看到自己玩了多长时间,增加刺激性.这个功能的核心有两点: 1. 时间计算:当前时刻- 游戏结束的时刻 2. 显示时间到窗口 时间计算 我们在初始化窗口时,就保存一个起始时间:当飞机死亡时,保存一个结束时间.我们在MyGameFrame中定义两个成员变量,如示例1所示. [示例1]定义时间变量

“全栈”工程师笔记/记一个完整的项目流程

引语:相信很多人都自认为自己是个全栈工程师,不管有没有验证过,我也不例外.心中总有一种傲气,事情都能做,只是做得好不好,时间够不够的问题!所以,对很多事情,我其实是一点不怕的,随着时间的推移,人总是应该要进步的,去做一些没做过的事,才对得起成长二字! 刚好上上个月,公司有一个新的项目需求,需要做一个全新的系统,但是看起来也不难,所以任务就交给了我,我可以说我是这个项目负责人吗?应该是可以的!但是,最开始就已经存在了一些坑,等着我去跳,就连最开始过需求的时候,我也不在场!不过,最终,项目也终于交到

10.2-全栈Java笔记:最全面的IO技术(二)

上节我们聊到「IO技术概念及入门」这节我们继续聊一下IO技术的分类. Java中流的概念细分 按流的方向分类: 输入流:数据流向是数据源到程序(InputStream.Reader结尾的流) 输出流:数据流向是程序到目的地(OutPutStream.Writer结尾的流) 按处理的数据单元: 字节流:按照字节读取数据(InputStream.OutputStream)命名上以stream结尾的流一般是字节流. 字符流:按照字符读取数据(Reader.Writer).命名上以Reader/Writ