请输入您要查询的百科知识:

 

词条 J2ME小游戏
释义

J2ME 小游戏即在消费类电子设备上运行的游戏,例如在蜂窝电话、可视电话、数字机顶盒、汽车导航系统、个人数字助理(PDA)和移动手持设备(MID)上运行的小游戏大多为J2ME小游戏。J2ME是一种高度优化的Java运行环境,是Java的组成部分,它主要针对消费类电子设备的,在此基础上设计出的游戏统称为J2ME 小游戏。

简介

一个J2ME的2D游戏技术DEMO源码

计划了很久了,准备作为J2ME的Game APIs的例子贴出来,无奈一直不得空,直到最近才简单地整理了一下,把它共享出来。

游戏特色

严格地讲,这个不能算作一个完整的游戏,没有自己的创意,只是简单地模仿了一个Flash的小游戏,当时是为了自己练习用的,Game Play也很简单,但包含了J2ME的Game包中所有的东西,作为一个Demo来讲,内容还是很充实的。

Gaming API分类

在介绍之前,先简单地提一下Gaming API中的几个类,包括Sprite,TiledLayer,LayerManager,Media.Player等。

Sprite:

偶尔看到过有些人把它翻译成精灵,我偷懒一下,就不翻译了,这个类用来表达游戏中的一个活动的角色,包括玩家控制的Player角色和非玩家控制的角色(NPC)。

TiledLayer:

这个是用来表达背景的类,其实从绘图的角度来看,Sprite和TiledLayer没有本质差别,只是将要画在屏幕上的一幅图像而已,因此在Game包中它们都是Layer类的子类。并且都能够从一幅图像方便地构造。

LayerManager:

这个是用来管理所有图像对象的类,通过把Sprite和TiledLayer加入其中,J2ME设备就知道如何来绘制它们了。

GameCanvas:

最后要提到的是画布,这是所有可视对象最终要表演的舞台。其实跟以前的Canvas没有本质的不同,同样是提供了一个Graphics接口来供把一些内容画上去而已,但增加了对玩家输入的处理,能够通过按键状态来直接读入玩家按键操作,相对更加简便了。

还利用到了Media包中的Player和ToneControl来播放音乐,没有音乐和声音的游戏是不可能出现的,呵呵。

操作指南

先简单地介绍一下这个游戏,玩家只能控制角色(一个端着网兜的小人)水平地移动,接住自然下落的小球就得分,积分到一定程度后,小球下落速度将加快,直到最高速为止;如果没接到小球,也有相应的惩罚,最终游戏会Game Over。

为了避免过于单调,玩家角色不是简单地平移,而是利用了Sprite的简单帧动画来让角色看上去有些动作。其实很方便的,只是在移动位置时更换一下图像就行了,Sprite提供有几个方法NextFrame(),PrevFrame()用来切换。

声音部分就更简单了,只是重复地播放一段预先写进去的音乐,来自Sun的WTK中的一个例子。

不过既然提到了它,就还是先简单地说一下吧,免得后面介绍其他部分时有些疑问。

J2ME中的声音部分非常简单,当然效果也不太好,所需的基本元素只有如下几个,一个内容部分的Byte系列;一个是播放器Player;还有一个是控制部分的ToneControl。代码示例如下:

tonePlayer = Manager.createPlayer(Manager.TONE_DEVICE_LOCATOR);

tonePlayer.setLoopCount(-1);

tonePlayer.realize();

ToneControl tc = (ToneControl)tonePlayer.getControl("javax.microedition.media.control.ToneControl");

tc.setSequence(mySequence);

当然还需要些初始化工作和异常处理,然后就可以通过tonePlayer.start()/close()方法来控制声音的播放和停止了。

具体请参见源码中CanvasGetBall.java中的createTonePlayer()方法。

锣鼓响了半天,主角也该出场了。这个游戏里的主角只有一个接球小人,不过角色还有一个跟它演对手戏的NPC,就是那只从天而降的小球了。这两个类都是Sprite的子类,小球因为有些自己的动作,同时实现了Runnable接口,能自主活动。不过也很简单,NPC嘛,一般来说都是相对弱智些,不然也没法玩了,谁的反应速度跟得上机器啊,再说了,NPC知道的信息也要多些:)。

先来看看主角吧,BallPlayer类就是我们的主角,其实非常简单,在所有的6个类中,除了记分用的Score外就数它最小了。提供一幅图像把它实例化后,就只能通过左右移动来控制了,额外的几个方法都是跟记分有关系的,先略过不提。

先来看看它的构造方法:

public BallPlayer(Image img, int fw, int fh) {

super(img, fw, fh);

this.step = fw / 2;

}

主要工作都由它的父类Sprite做了,给出一幅图像,这个图像是用PNG格式提供的,大家可能留意到不是一幅单一的图像,而是有点象帧动画中的几个关键帧,不错,的确如此,构造方法中的后两个参数就是告诉Sprite如何分割这幅图像的。这里整个Player共有6个关键帧,比较粗糙,呵呵,自己动手截屏做的:)

Field step是用来控制主角的移动步伐的,为了快一点,取了它身宽的一半。

接下来我们看看如何移动它,就是通过这样两个方法来左移和右移。

public void left() {

prevFrame();

if( (getX() - step) >= -12 ) {

move( -1 * step, 0);

}

}

public void right () {

nextFrame();

if((getX() + step) < canvas.getWidth()) {

move(step, 0);

}

}

留意一下,这里只管相对位移,主角的开始位置通过setPosition来设定,在运动过程中最好就不要直接设置位置了,增大计算量,要不就看起来动作不自然了。

接下来简单说一下配角--球。球的构造跟主角类似,只是为了节省构造销毁对象带来的开销,这个对象是一直存在的,也就是让它掉下去了又自己起来,并根据记分来确定下落速度,简单地用线程实现的,没怎么仔细设计,大家看看代码就清楚了。

再来看看CanvasGetBall这个类,它从GameCanvas继承,并实现了CommandListener和Runnable两个接口,是整个游戏中最复杂的一个类了,主要工作有如下几个部分,实例化主角,配角对象,还有背景对象,音乐等,并在适当的时候画出这些对象,在顶部画出些状态信息,并根据玩家操作开始和暂停游戏,并显示相应画面。并通过进行碰撞检测来判定玩家是否接到了小球。

检测方法

检测方法如下:

private boolean notMiss( ) {

// return player.collidesWith(ball,false);

int ballCX = ball.getX() + ball.getWidth()/2;

int ballCY = ball.getY() + ball.getHeight()/2;

int playerCX = player.getX() + player.getWidth()/2;

int playerCY = player.getY() + player.getHeight()/2;

return ((Math.abs(playerCX - ballCX)< ball.getWidth()/2) &&

(Math.abs(ballCY - playerCY) < 5));

}

被注释掉的一行是直接用Sprite的碰撞检测,下面的部分是自己计算两幅图像有没有重叠,效果差不多。其中collidesWith()的第二个参数是告诉内部方法是否要用像素级别的检测,通常答案是千万不要,这很慢的,而且没有必要这么精确。

为了说明整个游戏的控制逻辑,我们先来看看MIDletGetBall这个类,跟通常的MIDlet略有不同,因为我把主线程放在了CanvasGetBall中,MIDletGetBall只是简单地控制主线程就行了。

public void startMainThread() {

Display.getDisplay(this).setCurrent(displayable);

if(mainThread != null) {

mainThread = null;

Runtime.getRuntime().gc();

}

mainThread = new Thread(displayable);

mainThread.start();

}

其中第一行就是设置当前显示页面;也就是显示CanvasGetBall。

回到CanvasGetBall,整个游戏分几个阶段,相应有不同的画面和命令接口,详细说明如下:

1. 等待开始,对应在方法ready():

public void ready() {

cover.setTitle(TIPS[2]);

cover.addCommand(playCommand);

Display.getDisplay(MIDletGetBall.instance).setCurrent(cover);

}

为了绘制方便,这里单独用了个GameCanvas来绘制提示信息和响应命令,并根据玩家操作在CanvasCover和CanvasGetBall两个画面之间来回切换。

2. 游戏画面,包括启动和结束两个方法:

public void start() {

if(!playing) {

strTip = TIPS[0];

playing = true;

MIDletGetBall.instance.startMainThread();

removeCommand(playCommand);

removeCommand(resumeCommand);

addCommand(pauseCommand);

ball.start();

try {

if(tonePlayer != null) {

tonePlayer.start();

}

}

catch (MediaException ex) {

tonePlayer.close();

tonePlayer = null;

}

}

}

public void stop () {

if(playing) {

ball.stop();

strTip = TIPS[1];

try {

Thread.sleep(300);

}

catch (InterruptedException ex) {

}

playing = false;

removeCommand(pauseCommand);

addCommand(resumeCommand);

try {

tonePlayer.stop();

}

catch (MediaException ex1) {

tonePlayer.close();

tonePlayer = null;

}

}

}

并对应设置相应的命令来让玩家能够继续下去,构成一个简单的封闭控制环路。

3.游戏结束,对应方法gameover()

public void gameover() {

this.stop();

cover.setTitle(TIPS[3]);

cover.removeCommand(playCommand);

cover.addCommand(restartCommand);

Display.getDisplay(MIDletGetBall.instance).setCurrent(cover);

}

说到这里,基本上也就把它讲完了,具体内容请详细研究源码,其实没必要看太多书,深入地研究一个问题并根据自己的理解来改进或者是修正它,实践才是最好的老师,希望大家能够有所收获。

存在的问题

总结一下,这个游戏存在的问题有如下几个:

1. 没有好的Game Play,画面很差;

2. 可玩性不强,控制比较单调;

3. 游戏声音过于单调;

4. 运行速度有些慢。

但作为一个技术Demo,它涵盖了Game包中的所有内容,并提供了一个利用线程方式实现简单游戏的方法,很简单,但不适合真实的游戏,比较费时。

背景处理很差,可以通过一个Map来分割组合处理背景小片,能让游戏场景变得生动些,可以实现类似于卷轴游戏的效果,自己试试吧!

附:源代码和工程,在JBuilderX下编译,同时需要WTK2.0或以上版本。好久没有用Jbuilder了,买不起正版:),现在主要开发工具是Eclipse和EclipseME,感觉非常爽,免费的也有好货。

随便看

 

百科全书收录4421916条中文百科知识,基本涵盖了大多数领域的百科知识,是一部内容开放、自由的电子版百科全书。

 

Copyright © 2004-2023 Cnenc.net All Rights Reserved
更新时间:2025/1/26 15:51:29