Android 定时器实现的几种方式和removeCallbacks失效问题详解
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了Android 定时器实现的几种方式和removeCallbacks失效问题详解,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含3482字,纯文字阅读大概需要5分钟。
内容图文
![Android 定时器实现的几种方式和removeCallbacks失效问题详解](/upload/InfoBanner/zyjiaocheng/1180/d5e19e46e074402d824e434dc0cc30cd.jpg)
实现定时器有很多种方式,在这里我简单的介绍几种方式 ( 1)使用Handler + Runnable的方式 [java] view plain copy 在CODE上查看代码片派生到我的代码片 Handler handler = new Handler(); Runnable runnable = new Runnable() { @Override publicvoid run() { //你要做的事 //...... System.out.println(Thread.currentThread().getName()); handler.postDelayed(runnable, 1000); } }; 然后调用handler.post(runnable);就能启动定时器,这里是每隔1s打印线程名字,从打印中我们可以知道,他并没有另开线程,而是运行在UI线程当中,当你要取消定时器的时候,只需要调用handler.removeCallbacks(runnable)就可以了。 上面中有一个问题,有时候你会发现removeCallbacks有时候会失效,不能从消息队列中移除,看下面的demo
图:两个按钮,一个将Runnable加到消息队列中,一个将Runnable从消息队列中移除。该Runnable每1秒钟打印一次日志。 [java] view plain copy 在CODE上查看代码片派生到我的代码片 <span style="font-family:Courier New;">package com.example.demoactivity; import android.app.Activity; import android.os.Bundle; import android.os.Handler; import android.view.View; import android.view.View.OnClickListener; import android.widget.Button; publicclass TimerActivity extends Activity{ Handler handler = new Handler(); Runnable runnable = new Runnable() { @Override publicvoid run() { System.out.println("update..."); handler.postDelayed(runnable, 1000); } }; @Override protectedvoid onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.timer); Button mButtonStart = (Button) findViewById(R.id.button1); Button mButtonStop = (Button) findViewById(R.id.button2); mButtonStart.setOnClickListener(new OnClickListener() { @Override publicvoid onClick(View v) { handler.post(runnable); } }); mButtonStop.setOnClickListener(new OnClickListener() { @Override publicvoid onClick(View v) { handler.removeCallbacks(runnable); } }); } }</span><span style="font-family: Georgia, ‘Times new roman‘, Times, san-serif;"> </span> 结果: (1)start –> 输出 –> stop–> 停止输出 (2)start –> 输出 –> Background –> Front –> stop->继续输出 当Activity进入后台运行后再转入前台运行,removeCallbacks无法将updateThread从message queue中移除。 这是为什么呢? 在Activity由前台转后台过程中,线程是一直在运行的,但是当Activity转入前台时会重新定义Runnable runnable;也就是说此时从message queue移除的runnable与原先加入message queue中的runnable并非是同一个对象。如果把runnable定义为静态的则removeCallbacks不会失效,对于静态变量在内存中只有一个拷贝(节省内存),JVM只为静态分配一次内存,在加载类的过程中完成静态变量的内存分配,我们做如下修改就能解决上面的这个问题
static Handler handler = new Handler(); static Runnable runnable = new Runnable() { @Override publicvoid run() { System.out.println("update..."); handler.postDelayed(runnable, 1000); } }; (2)使用Timer的方式 [java] view plain copy 在CODE上查看代码片派生到我的代码片 Timer timer = new Timer(); timer.schedule(new TimerTask() { @Override publicvoid run() { System.out.println("update...."); } }, 0, 1000); 上面的每一秒打印语句,run方法是运行在子线程,不能直接在里面更新UI操作,这里需要注意下,取消的话调用timer.cancel()就能移除任务了 (3)采用Handle与线程的sleep(long )方法 1.定义一个Handler类,用于处理接受到的Message [java] view plain copy 在CODE上查看代码片派生到我的代码片 Handler handler = new Handler() { publicvoid handleMessage(Message msg) { super.handleMessage(msg); System.out.println("update..."); } } 2.新建一个实现Runnable接口的线程类,用一个boolean 来控制线程开始和结束 boolean isLive = true如下: [java] view plain copy 在CODE上查看代码片派生到我的代码片 publicclass MyThread implements Runnable { @Override publicvoid run() { while (isLive) { try { Thread.sleep(1000);// 线程暂停1秒,单位毫秒 Message message = new Message(); message.what = 1; handler.sendMessage(message);// 发送消息 } catch (InterruptedException e) { e.printStackTrace(); } } } } 3.在需要启动线程的地方加入下面语句 [java] view plain copy 在CODE上查看代码片派生到我的代码片 new Thread(new MyThread()).start(); 4.取消的话将isLive设置为false就行了 今天主要介绍这三种方法,写的不好的地方希望大家指出,谢谢!
原文:http://www.cnblogs.com/yaowen/p/5667530.html
内容总结
以上是互联网集市为您收集整理的Android 定时器实现的几种方式和removeCallbacks失效问题详解全部内容,希望文章能够帮你解决Android 定时器实现的几种方式和removeCallbacks失效问题详解所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。