入门-J2ME学习日记之-利用定时器类模拟MIDlet外部事件 作者:Snail
版权申明:可以转载,请保留作者信息和来源地址: 作者:Snail 地址:http://www.matrix.org.cn/resource/article/43/43851_J2ME.html
前面提到MIDlet程序本身可以通过调用notifyPaused()请求自己从活动状态进入暂停状态;调用notifyDestroyed()请求进入销毁状态;调用resumeRequest()请求恢复到活动状态。但是具体应该怎么使用呢?怎样通过程序本身模拟状态之间的转换呢?
这些都可以通过java.util包中的Timer 和TimerTask 类来实现。听说nokia的模拟器最接近真机,所以这次程序选择nokia s40 开发环境(唯一的不足就是不支持中文)。具体看如下演示程序:
import javax.microedition.midlet.MIDlet; import java.util.*; /* * 创建日期 2005-10-8 * * TODO 要更改此生成的文件的模板,请转至 * 窗口 - 首选项 - Java - 代码样式 - 代码模板 */
/** * @author Snail * * TODO 要更改此生成的类型注释的模板,请转至 * 窗口 - 首选项 - Java - 代码样式 - 代码模板 */ public class MyTimerTask extends TimerTask { private MIDlet midlet;
/** * */ public MyTimerTask(MIDlet midlet) { // TODO 自动生成构造函数存根 System.out.println("MyTimerTask contructor"); this.midlet = midlet; } public void run(){ System.out.println("run() called"); midlet.resumeRequest(); }
}
import javax.microedition.midlet.MIDlet; import javax.microedition.lcdui.*; import java.util.*; import javax.microedition.midlet.MIDletStateChangeException; /* * 创建日期 2005-10-8 * * TODO 要更改此生成的文件的模板,请转至 * 窗口 - 首选项 - Java - 代码样式 - 代码模板 */
/** * @author Snail * * TODO 要更改此生成的类型注释的模板,请转至 * 窗口 - 首选项 - Java - 代码样式 - 代码模板 */ public class MidletTest extends MIDlet implements CommandListener{ private Timer timer; private MyTimerTask mtk; private Command exit; /** * */ public MidletTest() { System.out.println("MidletTest Constructor"); //初始化Timer对象 timer = new Timer(); }
/* (非 Javadoc) * @see javax.microedition.midlet.MIDlet#startApp() */ protected void startApp() throws MIDletStateChangeException { // TODO 自动生成方法存根 System.out.println("startApp Called"); exit = new Command("EXIT", Command.EXIT, 1); Form f = new Form("MidletTest"); String s = new String("I'll come back soon!"); f.append(s); f.addCommand(exit); f.setCommandListener(this); Display.getDisplay(this).setCurrent(f); try{ //画面停留4秒 即活动状态 时间为4秒 Thread.sleep(4000); }catch(Exception e){} System.out.println("Ready to paused"); try{ mtk = null; //获得当前Midlet基类的引用(向上转型) mtk = new MyTimerTask(this); //执行该任务等待 2秒 timer.schedule(mtk, 2000); //请求进入暂停状态 pauseApp(); notifyPaused(); }catch(Exception e){} }
/* (非 Javadoc) * @see javax.microedition.midlet.MIDlet#pauseApp() */ protected void pauseApp() { // TODO 自动生成方法存根 System.out.println("pauseApp Called");
}
/* (非 Javadoc) * @see javax.microedition.midlet.MIDlet#destroyApp(boolean) */ protected void destroyApp(boolean arg0)throws MIDletStateChangeException { // TODO 自动生成方法存根 System.out.println("destroyApp Called:" + arg0); //停止Timer ,和TimerTask合同期满,脱离关系 timer.cancel();
} public void commandAction(Command c, Displayable d){ if(c == exit){ try{ destroyApp(false); notifyDestroyed(); }catch(MIDletStateChangeException e){} } }
}
通过上面程序可以看到, Timer必须配合TimerTask才能实现定时器功能。TimerTask是一个抽象类 ,必须有子类继承它并重载其run()方法。Timer实例一个对象timer,该对象通过调用schedule方法调度TimerTask子类对象执行其run()方法,从而达到使MIDlet 周期性的在暂停状态和活动状态之间不停的切换。
MyTimerTask类其实很简单,该类内部定义了一个私有数据 MIDlet 对象,通过它可以调用resumeRequest()方法使MIDlet 程序请求从暂停恢复到活动状态。
MidletTest 类中实现了CommandListener接口,所以必须要重载其对应的commandAction(Command c, Displayable d)方法。使程序能够对用户的操作做出响应。类似的在AWT中我们见多了。我们可以执行"EXIT",使程序主动进入销毁状态。具体可以观察控制台信息:
MidletTest Constructor startApp Called Ready to paused MyTimerTask contructor pauseApp Called run() called startApp Called destroyApp Called:false Ready to paused MyTimerTask contructor
便于理解,可以假设有如下场景:
上班时间,老板不在,用手机玩会游戏先!游戏ing…… 有电话!接个电话先,over! 继续游戏! 不好 b老板来了,"EXIT" 退出游戏!!
可以看到程序不停的在活动和暂停状态之间切换。当我们"EXIT"程序时 ,看到控制台打印信息 , 传入destroyApp 的是false,即非强制性销毁。前面已经了解,MIDlet主动请求状态转换需要调用的方法,并且一般都要先调用相应的pauseApp() destroyApp()方法。这里不在多说。现在关心的是:具体怎么通过Timer 和TimerTask 实现的呢?仔细观察程序,有这两句:
mtk = new MyTimerTask(this); timer.schedule(mtk, 2000);
首先实例MyTimerTask对象,并将this作为参数传递给构造方法。this指向的是当前Midlet。观察其构造方法,参数类型是MIDlet ,而MidletTest 只是其导出类。这里隐藏了向上转型。(具体可以参考j2se中继承与多态部分。)获得了MIDlet的引用 ,就象手里拿着遥控器一样 ,timer就可以控制MIDlet了。timer通过schedule方法设置周期,执行该任务 等待两秒。接下来的两句 ,就是程序主动请求进入暂停。另外,还注意到 timer 在构造方法中,所以只被调用一次,而mtk却每次调用startApp时都会被重新new一次。这是因为只有调用cancel() Timer和TimerTask脱离关系时 ,TimerTask才能被安排其他任务。
如果把Thread.sleep(4000); 这个try{}catch(){}块注释掉,我们发现 MIDlet从暂停进入活动状态后 屏幕只是很快的一闪 就又进入暂停状态了。所以这段代码 就是让活动状态多停留一会 让他sleep 4秒。
再看看这段代码:
try{ mtk = null; //获得当前Midlet基类的引用(向上转型) mtk = new MyTimerTask(this); //执行该任务等待 2秒 timer.schedule(mtk, 2000); //请求进入暂停状态 pauseApp(); notifyPaused(); }catch(Exception e){}
现在把其中的try 和catch 注释掉 其他的保留。当再次运行程序,"EXIT",看看控制台打印的信息:
MidletTest Constructor startApp Called destroyApp Called:false Ready to paused MyTimerTask contructor startApp threw an Exception java.lang.IllegalStateException: Timer already cancelled.
在startApp()中有异常被抛出。分析控制台信息,应该是发生在destroyApp已经被调用但notifyDestroyed()方法还没执行时。此时cancel()方法已经执行完毕,Timer和TimerTask 的合作关系也被迫解除。startApp()继续执行,当程序执行到
timer.schedule(mtk, 2000);
这句时 ,因为关系已经解除,startApp()无法继续, 所以抛出了上面的异常。
Timer类还有其他的方法,具体可以参考api中java.util包。
到此,就暂告一段落!旅行还将继续……
|