J2ME学习日记之利用定时器类模拟MIDlet外部事件

版权申明:可以转载,请保留作者信息和来源地址:
作者: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包