在进行Java游戏开发时,我们经常会遇到碰撞检测的问题。如坦克大战中,炮弹与坦克相遇发生爆炸;守卫者游戏中,守卫者发射的箭与怪物相遇使怪物失血;打飞机游戏中,飞机发送的子弹与敌机相遇干掉敌机。这些都需要实现碰撞检测。

我们先来看一类比较简单的碰撞检测:规则图形的碰撞检测。

矩形碰撞检测

作为一个练手的小游戏,游戏中的物体形状一般为矩形区域,这是规则图形。它的碰撞检测可以通过Java API中的Rectangle类来实现碰撞的检测。

Rectangle指的是一个矩形区域,它通过指定左上角位置x和y,以及矩形宽度和高度来确定范围大小。所以经常使用的 Rectangle类构造方法有:

// 构造一个新的 Rectangle,其左上角的坐标为 (0,0),其宽度和高度由同名的参数指定。
public Rectangle(int width, int height)

// 构造一个新的 Rectangle,其左上角被指定为 (x,y),其宽度和高度由同名的参数指定。
public Rectangle(int x, int y, int width, int height)

和碰撞检测相关的方法:

// 计算此 Rectangle 与指定 Rectangle 的交集
public Rectangle intersection(Rectangle r)

// 确定此 Rectangle 是否与指定的 Rectangle 相交
public boolean intersects(Rectangle r)

如果两个Rectangle对象有交集,那么他们就有碰撞了。如:

这种方法适用于地图中的物体近似为矩形或者虽然不是矩形,但是碰撞精度要求不高的情况下的碰撞检测。每个物体记录一个能够将自己框住的最小矩形的左上角坐标和矩形长宽。

采用此种方法进行碰撞检测需要注意,对于图片的实现处理应该尽量的去掉图标边角的空白,不然实际效果可以产生肉眼可辨的误差。也就是说Rectangle尽量的包住图形且Rectangle的区域尽量小。

示例:

import java.awt.Graphics;
import java.awt.Image;
import java.awt.MediaTracker;
import java.awt.Point;
import java.awt.Rectangle;
import java.awt.Toolkit;
import java.awt.image.CropImageFilter;
import java.awt.image.FilteredImageSource;
import java.awt.image.ImageFilter;
import java.awt.image.ImageProducer;

import javax.swing.JFrame;
import javax.swing.JOptionPane;

/**
* 碰撞检测测试,判断两个同方向移动的坦克是否会发生碰撞
*
* @author 
*
*/
public class Intersection extends JFrame implements Runnable {

private static final long serialVersionUID = 156638225301569550L;
private MediaTracker mediaTracker; // 媒体追踪器
private Image[][] images = new Image[2][4]; // 放置所有切割的图像
private Image[] moveTanks = new Image[2]; // 放置界面移动的两辆坦克
private Point[] points = new Point[2]; // 两辆坦克坐标

public Intersection() {
setTitle("碰撞检测");
setSize(200, 600);
setLocationRelativeTo(null);
setDefaultCloseOperation(EXIT_ON_CLOSE);

cutImage(); // 切割图形
/* 将切割的图形添加到媒体追踪器中跟踪 */
mediaTracker = new MediaTracker(this);
for (int i = 0, length = images.length; i < length; i++) {
for (int j = 0, len = images[i].length; j < len; j++) {
mediaTracker.addImage(images[i][j], i * len + j);
}
}

// 等待所有图像加载完毕
try {
mediaTracker.waitForAll();
} catch (InterruptedException e) {
e.printStackTrace();
}

// 初始化两辆坦克在窗体中坐标
moveTanks[0] = createImage(images[0][0].getSource());
moveTanks[1] = createImage(images[0][0].getSource());
points[0] = new Point(80, 200);
points[1] = new Point(80, 100);

setVisible(true);
}

/**
* 图像分割
*/
private void cutImage() {
// 获取源图像
Image img = Toolkit.getDefaultToolkit().getImage("images/boss.gif");
// 循环分割图像
for (int i = 0, length = images.length; i < length; i++) {
for (int j = 0, len = images[i].length; j < len; j++) {
ImageFilter filter = new CropImageFilter(0, 0, 50, 50);
ImageProducer producer = new FilteredImageSource(
img.getSource(), filter);
images[i][j] = createImage(producer); // 将分割后图像放入数组中保存
}
}
}

@Override
public void paint(Graphics g) {
Image img = createImage(this.getWidth(), this.getHeight());
Graphics graphics = img.getGraphics();
// 将两辆坦克在窗体中绘制出来
for (int i = 0, len = moveTanks.length; i < len; i++) {
graphics.drawImage(moveTanks[i], points[i].x, points[i].y, this);
}
g.drawImage(img, 0, 0, this);
g.dispose();
}

@Override
public void run() {
while (true) {
// 每次第二辆坦克移动距离比第一辆坦克大,即第二辆移动更快一些
points[0].y += 30;
points[1].y += 45;
// 重绘
repaint();

/* 碰撞检测 */
// 第一辆坦克的矩形范围
Rectangle tank1 = new Rectangle(points[0].x, points[0].y,
moveTanks[0].getWidth(null), moveTanks[0].getHeight(null));
// 第二辆坦克的矩形范围
Rectangle tank2 = new Rectangle(points[1].x, points[1].y,
moveTanks[1].getWidth(null), moveTanks[1].getHeight(null));
// 判断两个矩形是否有交集,有则说明碰撞了
if (tank1.intersects(tank2)) {
JOptionPane.showMessageDialog(null, "碰到了", "提示",
JOptionPane.INFORMATION_MESSAGE);
break;
}

try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

public static void main(String[] args) {
new Thread(new Intersection()).start();
}
}


圆形碰撞检测

圆形检测与矩形检测方法类似,区别在于用一个能够包含物体的圆代替了矩形。主要是考虑到游戏中的物体外形以平滑为主,例如人物角色。而判断两个圆是否碰撞的计算也很简单,就是判断两个圆心之间的距离是否小于两个圆的半径之和。

示例:

import java.awt.Graphics;

import javax.swing.JFrame;
import javax.swing.JOptionPane;

/**
* 碰撞检测测试,判断两个圆是否会发生碰撞
*
* @author 
*
*/
public class Intersection2 extends JFrame implements Runnable {
/* 定义两圆左上角坐标、半径 */
private int x1 = 15, y1 = 45;
private int x2 = 35, y2 = 70;
private int r1 = 16, r2 = 18;

public Intersection2() {
setTitle("碰撞检测");
setSize(200, 600);
setLocationRelativeTo(null);
setDefaultCloseOperation(EXIT_ON_CLOSE);

setVisible(true);
}

@Override
public void paint(Graphics g) {
/* 画圆 */
g.drawOval(x1, y1, 2 * r1, 2 * r1);
g.drawOval(x2, y2, 2 * r2, 2 * r2);
}

@Override
public void run() {
/* 判断两圆是否相交 */
// 两圆中心坐标
int centerX1 = x1 + r1, centerY1 = y1 + r1;
int centerX2 = x2 + r2, centerY2 = y2 + r2;
// 求两圆的圆心距
double length = Math.sqrt(Math.pow(centerX1 - centerX2, 2)
+ Math.pow(centerY1 - centerY2, 2));
// 判断圆心距与两圆半径和的关系
if (length < (r1 + r2)) {
JOptionPane.showMessageDialog(null, "圆心距:" + length + ",碰撞了");
} else {
JOptionPane.showMessageDialog(null, "圆心距:" + length + ",未碰撞");
}
}

public static void main(String[] args) {
new Thread(new Intersection2()).start();
}
}

package state;

import java.awt.Graphics;
import java.awt.Image;
import java.awt.MediaTracker;
import java.awt.Point;
import java.awt.Rectangle;

import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JOptionPane;

public class Intersection extends JFrame implements Runnable{

private MediaTracker mediaTracker;
private Image[] images = new Image[2];
private Point[] point = new Point[2];
private Image image;

public Intersection() {
setTitle("卡片布局管理器");
setSize(500, 500);
setLocationRelativeTo(null);
setResizable(false);
setDefaultCloseOperation(EXIT_ON_CLOSE);
mediaTracker = new MediaTracker(this);
images[0] = ImageUtils.addImage(mediaTracker, new ImageIcon("imges/jian_super.png").getImage());
point[0] = new Point(30, 100);
images[1] = ImageUtils.addImage(mediaTracker, new ImageIcon("imges/xiaoguai.png").getImage());
point[1] = new Point(400, 100);

setVisible(true);

}

@Override
public void paint(Graphics g) {
g.drawImage(image, 0, 0, this);
}

private void draw(){
image = createImage(this.getWidth(), this.getHeight());
Graphics g = image.getGraphics();
for(int i =0 ;i<images.length;i++){
g.drawImage(images[i], point[i].x, point[i].y, this);
}
g.dispose();
}

private boolean intersects() {
Rectangle arrow = new Rectangle(point[0].x, point[0].y,
images[0].getWidth(null)-20, images[0].getHeight(null));
Rectangle boss = new Rectangle(point[1].x, point[1].y,
images[1].getWidth(null), images[1].getHeight(null));
return arrow.intersects(boss);

}

@Override
public void run() {
while(true){
point[0].x++;
point[1].x--;
if(intersects()){
JOptionPane.showMessageDialog(null, "碰撞");
break;
}
draw();
repaint();
try {
Thread.sleep(4);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

public static void main(String[] args) {
new Thread(new Intersection()).start();
}
}

Java游戏之碰撞检测的更多相关文章

  1. 【开源java游戏框架libgdx专题】-01-libgdx介绍

    libgdx是一款开源的java游戏框架,而且还实现了Desktop/Android/BlackBerry/iOS/HTML5这些些平台的跨平台开发.官方网址:https://libgdx.badlo ...

  2. java游戏开发杂谈 - 游戏物体

    现实生活中,有很多物体,每个物体的长相.行为都不同. 物体存在于不同的空间内,它只在这个空间内发生作用. 物体没用了,空间就把它剔除,不然既占地方,又需要花精力管理. 需要它的时候,就把它造出来,不需 ...

  3. java游戏开发杂谈 - 有限状态机

    在不同的阶段,游戏所运行的逻辑.所显示的界面,都是不同的. 以五子棋举例,游戏开始.游戏中.胜负已分,对应的界面和逻辑都不同. 在游戏中,又分为:自己下棋.对方下棋.游戏暂停.悔棋等多个状态. 再比如 ...

  4. java游戏开发杂谈 - 线程

    线程,让游戏拥有了动态变化的能力. java的图形界面,在启动的时候,就开始了一个线程. 这个线程负责处理:JFrame.JPanel等的绘制.事件处理. 它是由操作系统调用的,在程序启动时开启,程序 ...

  5. java游戏开发杂谈 - 实现游戏主菜单

    经常玩游戏的同学,大家都知道,游戏都会有个主菜单,里面有多个菜单选项:开始游戏.游戏设置.关于游戏.退出游戏等等,这个菜单是怎么实现的呢. 有一定桌面软件开发基础的同学可能会想到,用JButton组件 ...

  6. java游戏开发杂谈 - 事件处理

    大家都知道,游戏需要跟玩家交互,需要接收玩家的鼠标.键盘发出的命令,比如在地图上点击一下,人物就自动寻路走过去:键盘上按下某个键,就弹出一个背包界面. 这些逻辑是怎么处理的呢? 大家先不用深究太详细的 ...

  7. java游戏开发杂谈 - 画布和画笔

    在Eclipse里,编写如下两个类: package game2; import java.awt.Color; import java.awt.Graphics; import javax.swin ...

  8. java游戏开发杂谈 - 创建一个窗体

    package game1; import javax.swing.JFrame; /** * java游戏开发杂谈 * ---demo1:创建一个窗体 * * @author 台哥 * @date ...

  9. java游戏开发杂谈 - 游戏编程浅析

    每个游戏,你所看到的它的一切,都是计算机画出来的! 地图是画出来,人物是画出来的,树木建筑是画出来的,菜单按钮是画出来的,滚动的文字.闪烁的图标.云雾烟火,都是画出来的. 游戏编程,所要做的,就是控制 ...

随机推荐

  1. Python Pandas库的学习(二)

    今天我们继续讲下Python中一款数据分析很好的库.Pandas的学习 接着上回讲到的,如果有人听不懂,麻烦去翻阅一下我前面讲到的Pandas学习(一) 如果我们在数据中,想去3,4,5这几行数据,那 ...

  2. 集训第六周 数学概念与方法 概率 F题

    Submit Status Description Sometimes some mathematical results are hard to believe. One of the common ...

  3. jQuery学习之------元素样式的操作

    jQuery学习之------元素样式的操作 一..addClass( className )方法----增加样式 1.addClass( className ) : 为每个匹配元素所要增加的一个或多 ...

  4. bash shell & front-end & auto publish & auto deploy

    bash shell & front-end & auto publish & auto deploy $ zip -r apitool-2018-11-22.zip apit ...

  5. [bzoj4826][Hnoi2017]影魔_单调栈_主席树

    影魔 bzoj-4826 Hnoi-2017 题目大意:给定一个$n$个数的序列$a$,求满足一下情况的点对个数: 注释:$1\le n,m\le 2\cdot 10^5$,$1\le p1,p2\l ...

  6. HDU——1498 50 years, 50 colors

    50 years, 50 colors Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Othe ...

  7. int *ptr=(int *)(&a+1)问题的探讨

    从网络上看到这样一道有意思的题目,是关于数组与指针的问题,描述如下: main() { ]={,,,,}; ); printf(),*(ptr-)); } 输出为:2,5 请解释以上代码的输出结果. ...

  8. javaweb开发页面数字过长显示科学计数法的问题

    1. 检查该字段是否为double类型,如果是,请改成BigDecimal 2.如果是导出excel里面为科学计数法,原页面正常,是因为excel设置的原因,请参考https://jingyan.ba ...

  9. io计算

    http://www.cnblogs.com/yalong_xiang/archive/2011/11/15/2249530.html ぬ儱←OWEN★   windows下如何查看磁盘IO性能 复制 ...

  10. 一步步搭建java信息管理系统00 - 前言

    开发前,先上效果图吧 信息管理系统,个人认为,以下几个因素是不可缺少的 多tab 因菜单比较多,右侧的树形一定要考虑,如果菜单还是多,那么顶部就要考虑起来了 以后想到什么,再添加吧. 看到easyui ...