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

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

(1)使用Handler + Runnable的方式

[java] view plain copy
在CODE上查看代码片派生到我的代码片

    Handler handler = new Handler();  
    Runnable runnable = new Runnable() {  
          
        @Override  
        public void 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;  
      
    public class TimerActivity extends Activity{  
        Handler handler = new Handler();  
        Runnable runnable = new Runnable() {  
              
            @Override  
            public void run() {  
                System.out.println("update...");  
                handler.postDelayed(runnable, 1000);  
            }  
        };  
      
        @Override  
        protected void 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  
                public void onClick(View v) {  
                    handler.post(runnable);  
                }  
            });  
              
            mButtonStop.setOnClickListener(new OnClickListener() {  
                  
                @Override  
                public void 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  
        public void 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  
        public void 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() {  
            public void handleMessage(Message msg) {  
                super.handleMessage(msg);  
                System.out.println("update...");  
            }  
        }  

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

[java] view plain copy
在CODE上查看代码片派生到我的代码片

    public class MyThread implements Runnable {  
            @Override  
            public void 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开发
flutter中实现仿Android端的onResume和onPause方法
flutter中实现仿Android端的onResume和onPause方法
|
编解码 Android开发
性能优化:Android中Bitmap内存大小优化的几种常见方式
性能优化:Android中Bitmap内存大小优化的几种常见方式
|
Android开发 容器
Android实现面包屑效果,支持Fragment联动
Android实现面包屑效果,支持Fragment联动
|
Android开发
Android实现连线题效果
Android实现连线题效果
|
Android开发
Android实现调用系统相机录像及实现录音
Android实现调用系统相机录像及实现录音
581 0
|
移动开发 JavaScript Android开发
通过howler.js实现在Android下的微信浏览器自动播放音频
通过howler.js实现在Android下的微信浏览器自动播放音频
399 0
通过howler.js实现在Android下的微信浏览器自动播放音频
|
存储 Dart Java
【Flutter】packages思维以及使用Java添加Android平台特定的实现在Flutter框架里的体现和运用
【Flutter】packages思维以及使用Java添加Android平台特定的实现在Flutter框架里的体现和运用
|
缓存 JSON Java
java 实现读取txt文件,反射创建对象,android 手机缓存文件目录
java 实现读取txt文件,反射创建对象,android 手机缓存文件目录
335 1
java 实现读取txt文件,反射创建对象,android 手机缓存文件目录
|
数据库 Android开发
android 多级下拉菜单实现教程 greendao使用
android 多级下拉菜单实现教程 greendao使用
169 0
android 多级下拉菜单实现教程 greendao使用
|
XML 开发工具 Android开发
Android自定义控件(十三)——实现CSDN搜索框文字提示容器
Android自定义控件(十三)——实现CSDN搜索框文字提示容器
264 0
Android自定义控件(十三)——实现CSDN搜索框文字提示容器