最近在做一个GPS的应用,其中界面使用Fragment实现切换。其中一个Fragment中实现了数字时钟,即每一秒就更新数字显示那种~~
结果在切换Fragment时,出现了下面的错误:
01-23 14:54:19.891: D/AndroidRuntime(32574): Shutting down VM
01-23 14:54:19.891: E/AndroidRuntime(32574): FATAL EXCEPTION: main
01-23 14:54:19.891: E/AndroidRuntime(32574): Process: com.mobisummer.msgps, PID: 32574
01-23 14:54:19.891: E/AndroidRuntime(32574): java.lang.IllegalStateException: Fragment NumbersFragment{dab2ec6} not attached to Activity
01-23 14:54:19.891: E/AndroidRuntime(32574): at android.app.Fragment.getResources(Fragment.java:802)
01-23 14:54:19.891: E/AndroidRuntime(32574): at android.app.Fragment.getString(Fragment.java:824)
01-23 14:54:19.891: E/AndroidRuntime(32574): at com.mobisummer.msgps.NumbersFragment.setDateAndTime(NumbersFragment.java:197)
01-23 14:54:19.891: E/AndroidRuntime(32574): at com.mobisummer.msgps.NumbersFragment.access$0(NumbersFragment.java:194)
01-23 14:54:19.891: E/AndroidRuntime(32574): at com.mobisummer.msgps.NumbersFragment$1.handleMessage(NumbersFragment.java:73)
01-23 14:54:19.891: E/AndroidRuntime(32574): at android.os.Handler.dispatchMessage(Handler.java:102)
01-23 14:54:19.891: E/AndroidRuntime(32574): at android.os.Looper.loop(Looper.java:145)
01-23 14:54:19.891: E/AndroidRuntime(32574): at android.app.ActivityThread.main(ActivityThread.java:5972)
01-23 14:54:19.891: E/AndroidRuntime(32574): at java.lang.reflect.Method.invoke(Native Method)
01-23 14:54:19.891: E/AndroidRuntime(32574): at java.lang.reflect.Method.invoke(Method.java:372)
01-23 14:54:19.891: E/AndroidRuntime(32574): at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1399)
01-23 14:54:19.891: E/AndroidRuntime(32574): at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:1194)
一开始以为是Fragment的问题,但慢慢分析日志,发现开始出现问题是在时钟实现的过程中,那就必须回看时钟实现的代码:
private void updateDateAndTime() {
if (timer == null) {
timer = new Timer(true);
}
if (timer != null) {
if (timerTask != null) {
timerTask.cancel();
}
timerTask = new MyTimerTask();
timer.schedule(timerTask, 0, 1000);
}
}
private Handler mHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
// TODO Auto-generated method stub
super.handleMessage(msg);
if (msg.what == 1) {
//更新时间
setDateAndTime();
}
}
};
class MyTimerTask extends TimerTask {
@Override
public void run() {
// TODO Auto-generated method stub
Message msg = new Message();
msg.what = 1;
mHandler.sendMessage(msg);
}
}
实现时钟也是很简单,就是运用一个定时器Timer,定时一秒就执行一次任务TimerTask,任务TimerTask就是通过Handler通知UI线程更新时钟显示。
其实Timer就是一个线程,使用schedule方法完成对TimerTask的调度,多个TimerTask可以共用一个Timer,也就是说Timer对象调用一次schedule方法就是创建了一个线程,并且调用一次schedule 后TimerTask是无限制的循环下去的,使用Timer的cancel()停止操作。当然同一个Timer执行一次cancel()方法后,所有Timer线程都被终。
可见,上面我们还中cancel掉TimerTask 和Timer,否则该线程实例一直运行,当然会出错了。
所以添加下面的代码:
private void cancelTimer() {
timerTask.cancel();
timerTask = null;
timer.cancel();
timer = null;
}
到此,完成。当然实现数字时钟的其他细节就不一一讲述了,下一节,我会讲到如何实现LED数字时钟的“8”字显示