android定时器的使用

2018-03-26  本文已影响504人  努力深耕Android的小透明

  目前在项目中使用过的定时器主要是两种方式:1.handler的延时任务 2.timer的定时器

  下面主要来介绍这两种方式 定时器的用法,和遇到的坑 :
  1.handler的延时任务
  写了一个demo,是关于hanlder发送延时消息和 取消延时消息的


image.png

package com.phicomm.android.test.demo;

import android.os.Handler;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ListView;
import android.widget.Toast;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    private Button mOpenButton;
    private Handler mHandler;
    private Runnable mRunnable;
    private Button mCloseButton;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        initView();

    }

    private void initView() {
        mOpenButton = (Button) findViewById(R.id.btn_open);
        mCloseButton = (Button) findViewById(R.id.btn_close);
        mOpenButton.setOnClickListener(this);
        mCloseButton.setOnClickListener(this);
        mHandler = new Handler();

    }



    @Override
    public void onClick(View view) {
        switch (view.getId()){
            case R.id.btn_open:
                mRunnable = new Runnable() {
                    @Override
                    public void run() {
                        Toast.makeText(MainActivity.this, "1s后延时任务执行了"+mRunnable.toString(), Toast.LENGTH_SHORT).show();
                    }
                };
                mHandler.postDelayed(mRunnable,1000);
            break;
            case R.id.btn_close:
               mHandler.removeCallbacksAndMessages(mRunnable);
                break;
        }
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        if (mHandler != null) {
            mHandler.removeCallbacksAndMessages(mRunnable);
        }
    }
}


  但是在实际运用的过程中,常常会出现无法移除延时消息的问题,这个问题主要是因为两个runnable的对象不是同一个,
  当Activity进入后台运行后再转入前台运行,removeCallbacks无法将updateThread从message queue中移除。这是为什么呢?
  在Activity由前台转后台过程中,线程是一直在运行的,但是当Activity转入前台时会重新定义Runnable runnable;也就是说此时从message queue移除的runnable与原先加入message queue中的runnable并非是同一个对象。  解决方式1:如果把runnable定义为静态的则removeCallbacks不会失效,对于静态变量在内存中只有一个拷贝(节省内存),JVM只为静态分配一次内存,在加载类的过程中完成静态变量的内存分配,我们做如下修改就能解决上面的这个问题.

  解决方式2:将所有的runnable任务放到一个集合中,根据runnable的id移除相应的任务,即可以解决这个问题

2.timer的延时任务
同样用上面的demo来说明使用方法:
要注意的是:
TimerTask运行在一个单独的线程里,而不是UI线程。所以使用Android timer时,注意android的单线程原则,确保线程安全。不要在TimerTask的run方法中做UI相关的操作,如:TextView.setText()等,这样可能会导致UI线程阻塞。如果需要可以使用handler向UI线程发消息,具体处理由UI线程自己完成。在使用完Timer之后,要使用Timer的cancel方法取消Timer,否则Timer一直在运行。

image.png
package com.phicomm.android.test.timer;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import com.phicomm.android.test.demo.R;

import java.util.Timer;
import java.util.TimerTask;

public class TimerActivity extends AppCompatActivity implements View.OnClickListener {

    private Button mOpenButton;
    private Button mCloseButton;
    private Timer mTimer;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_timer);
        initView();
    }

    private void initView() {
        mOpenButton = (Button) findViewById(R.id.btn_open);
        mCloseButton = (Button) findViewById(R.id.btn_close);
        mOpenButton.setOnClickListener(this);
        mCloseButton.setOnClickListener(this);
        mTimer = new Timer();
    }

    @Override
    public void onClick(View view) {
        switch (view.getId()){
            case R.id.btn_open:
                TimerTask timerTask = new TimerTask(){
                    @Override
                    public void run() {
                       runOnUiThread(new Runnable() {
                           @Override
                           public void run() {
                               Toast.makeText(TimerActivity.this, "1s后延时任务执行了", Toast.LENGTH_SHORT).show();
                           }
                       });
                    }
                };
                mTimer.schedule(timerTask,1000);
                break;
            case R.id.btn_close:
                mTimer.cancel();

                break;
        }
    }
}

  这里需要注意的是,timer的cancel方法只能执行一次,不能在调用了cancel之后再进行timer的任务执行。

参考文章:
Android 定时器实现的几种方式和removeCallbacks失效问题详解

Android 中Timer和TimeTask完成定时任务

上一篇下一篇

猜你喜欢

热点阅读