原文网址:http://blog.csdn.net/xiaanming/article/details/9011193

实现定时器有很多种方式,在这里我简单的介绍几种方式

(1)使用Handler + Runnable的方式

  1. Handler handler = new Handler();
  2. Runnable runnable = new Runnable() {
  3. @Override
  4. public void run() {
  5. //你要做的事
  6. //......
  7. System.out.println(Thread.currentThread().getName());
  8. handler.postDelayed(runnable, 1000);
  9. }
  10. };

然后调用handler.post(runnable);就能启动定时器,这里是每隔1s打印线程名字,从打印中我们可以知道,他并没有另开线程,而是运行在UI线程当中,当你要取消定时器的时候,只需要调用handler.removeCallbacks(runnable)就可以了。

上面中有一个问题,有时候你会发现removeCallbacks有时候会失效,不能从消息队列中移除,看下面的demo

图:两个按钮,一个将Runnable加到消息队列中,一个将Runnable从消息队列中移除。该Runnable每1秒钟打印一次日志。

  1. <span style="font-family:Courier New;">package com.example.demoactivity;
  2. import android.app.Activity;
  3. import android.os.Bundle;
  4. import android.os.Handler;
  5. import android.view.View;
  6. import android.view.View.OnClickListener;
  7. import android.widget.Button;
  8. public class TimerActivity extends Activity{
  9. Handler handler = new Handler();
  10. Runnable runnable = new Runnable() {
  11. @Override
  12. public void run() {
  13. System.out.println("update...");
  14. handler.postDelayed(runnable, 1000);
  15. }
  16. };
  17. @Override
  18. protected void onCreate(Bundle savedInstanceState) {
  19. super.onCreate(savedInstanceState);
  20. setContentView(R.layout.timer);
  21. Button mButtonStart = (Button) findViewById(R.id.button1);
  22. Button mButtonStop = (Button) findViewById(R.id.button2);
  23. mButtonStart.setOnClickListener(new OnClickListener() {
  24. @Override
  25. public void onClick(View v) {
  26. handler.post(runnable);
  27. }
  28. });
  29. mButtonStop.setOnClickListener(new OnClickListener() {
  30. @Override
  31. public void onClick(View v) {
  32. handler.removeCallbacks(runnable);
  33. }
  34. });
  35. }
  36. }</span><span style="font-family: Georgia, 'Times new roman', Times, san-serif;">
  37. </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只为静态分配一次内存,在加载类的过程中完成静态变量的内存分配,我们做如下修改就能解决上面的这个问题

“在Activity由前台转后台过程中,线程是一直在运行的,但是当Activity转入前台时会重新定义Runnable runnable”应该不会重新定义Runnable runnable。

  1. static Handler handler = new Handler();
  2. static Runnable runnable = new Runnable() {
  3. @Override
  4. public void run() {
  5. System.out.println("update...");
  6. handler.postDelayed(runnable, 1000);
  7. }
  8. };

(2)使用Timer的方式

  1. Timer timer = new Timer();
  2. timer.schedule(new TimerTask() {
  3. @Override
  4. public void run() {
  5. System.out.println("update....");
  6. }
  7. }, 0, 1000);

上面的每一秒打印语句,run方法是运行在子线程,不能直接在里面更新UI操作,这里需要注意下,取消的话调用timer.cancel()就能移除任务了

(3)采用Handle与线程的sleep(long )方法

1.定义一个Handler类,用于处理接受到的Message

  1. Handler handler = new Handler() {
  2. public void handleMessage(Message msg) {
  3. super.handleMessage(msg);
  4. System.out.println("update...");
  5. }
  6. }

2.新建一个实现Runnable接口的线程类,用一个boolean 来控制线程开始和结束  boolean isLive = true如下:

  1. public class MyThread implements Runnable {
  2. @Override
  3. public void run() {
  4. while (isLive) {
  5. try {
  6. Thread.sleep(1000);// 线程暂停1秒,单位毫秒
  7. Message message = new Message();
  8. message.what = 1;
  9. handler.sendMessage(message);// 发送消息
  10. } catch (InterruptedException e) {
  11. e.printStackTrace();
  12. }
  13. }
  14. }
  15. }

3.在需要启动线程的地方加入下面语句

  1. new Thread(new MyThread()).start();

4.取消的话将isLive设置为false就行了

今天主要介绍这三种方法,写的不好的地方希望大家指出,谢谢!

【转】Android 定时器实现的几种方式和removeCallbacks失效问题详解--不错的更多相关文章

  1. Android 定时器实现的几种方式和removeCallbacks失效问题详解

    实现定时器有很多种方式,在这里我简单的介绍几种方式 (1)使用Handler + Runnable的方式 Handler handler = new Handler(); Runnable runna ...

  2. Form_Form页面跳转的四种方式(open_form, call_form, new_form, fnd_function)详解(汇总)

    2014-06-29 Created By BaoXinjian

  3. uni-app&H5&Android混合开发三 || uni-app调用Android原生方法的三种方式

    前言: 关于H5的调用Android原生方法的方式有很多,在该片文章中我主要简单介绍三种与Android原生方法交互的方式. 一.H5+方法调用android原生方法 H5+ Android开发规范官 ...

  4. JAVA高级架构师基础功:Spring中AOP的两种代理方式:动态代理和CGLIB详解

    在spring框架中使用了两种代理方式: 1.JDK自带的动态代理. 2.Spring框架自己提供的CGLIB的方式. 这两种也是Spring框架核心AOP的基础. 在详细讲解上述提到的动态代理和CG ...

  5. Android开发之手势滑动(滑动手势监听)详解

    Android开发之手势滑动(滑动手势监听)详解 在Android应用中,经常需要手势滑动操作,比如上下滑动,或左右方向滑动,处理手势滑动通常有两种方法:一种是单独实现setOnTouchListen ...

  6. jQuery插件开发的两种方法及$.fn.extend的详解(转)

    jQuery插件开发的两种方法及$.fn.extend的详解 jQuery插件开发分为两种:1 类级别.2 对象级别,下面为大家详细介绍下   jQuery插件开发分为两种: 1 类级别 类级别你可以 ...

  7. Android Studio一直显示Building“project name”Gradle project info问题详解

    关注我,每天都有优质技术文章推送,工作,学习累了的时候放松一下自己. 本篇文章同步微信公众号  欢迎大家关注我的微信公众号:「醉翁猫咪」 Android Studio一直显示 Building&quo ...

  8. Android 更新UI的几种方式

    1.Activity的 runOnUiThread textView = (TextView) findViewById( R.id.tv ); new Thread(new Runnable() { ...

  9. Android请求服务器的两种方式--post, get的区别

    android中用get和post方式向服务器提交请求_疯狂之桥_新浪博客http://blog.sina.com.cn/s/blog_a46817ff01017yxt.html Android提交数 ...

随机推荐

  1. POJ-2386(深广搜基础)

    Lake Counting Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 25322   Accepted: 12759 D ...

  2. 对象拷贝类PropertyUtils,BeanUtils,BeanCopier的技术沉淀

    功能简介 对象拷贝的应用现状简介: 业务系统中经常需要两个对象进行属性的拷贝,不能否认逐个的对象拷贝是最快速最安全的做法,但是当数据对象的属性字段数量超过程序员的容忍的程度,代码因此变得臃肿不堪,使用 ...

  3. js 截取字符串

    转:http://blog.csdn.net/dotnet25/article/details/8331959 字符串:var s = "1,2,3,4,5," 目标:删除最后一个 ...

  4. poj 1089 Intervals

    http://poj.org/problem?id=1089 Intervals Time Limit: 1000MS   Memory Limit: 10000K Total Submissions ...

  5. js中的call()与apply()

    js中的call()函数和apply()函数: 1.主要作用:是用于指定作用域和传参 (1)用于指定作用域 window.color = "red"; var o = { colo ...

  6. 每天进步一点点——Linux系统时间来处理

    转载请注明出处:http://blog.csdn.net/cywosp/article/details/25839551 在程序中时间处理往往是一个麻烦的事.Linux系统提供了非常多关于时间处理的函 ...

  7. linux 远程自动登录脚本 (注test.exp)

    #! /usr/bin/expect set timeout 30spawn ssh -l root 192.168.239.148 expect "password:"send ...

  8. tstring

      是的,一旦知道 TCHAR 和_T 是如何工作的,那么这个问题很简单.基本思想是 TCHAR 要么是char,要么是 wchar_t,这取决于 _UNICODE 的值: // abridged f ...

  9. CCSpawn使用CCRepeatForever无效

    在使用CCSpawn中加入CCRepeatForever,但却无法使用CCRepeatForever的效果. CCActionInterval* action =(CCActionInterval*) ...

  10. SAN和NAS的区别

    SAN : STORAGE AREA NETWORK   存储区域网络 NAS : NETWORK ATTACHED STORAGE 网络附加存储 NAS不一定是盘阵,一台普通的主机就可以做出NAS, ...