Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android計時器的三種實現方式(Chronometer、Timer、handler)

Android計時器的三種實現方式(Chronometer、Timer、handler)

編輯:關於Android編程

本文實例為大家分享了Android計時器的三種方法,具體內容如下

目錄:

1、借助Timer實現

2、調用handler.sendMessagedely(Message msg, long delayMillis)

3、借助布局Chronometer

1、借助Timer實現

(1) 布局文件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="match_parent"
  android:layout_height="match_parent"
  android:orientation="vertical" >
 
  <TextView
   android:id="@+id/timerView"
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:layout_gravity="center_horizontal"
   android:textSize="60sp" />
 
 </LinearLayout>

布局文件很簡單,就是一個TextView用來顯示計時時間。下面看一下Activity裡的邏輯實現:

(2)Activity文件

public class MyChronometer extends Activity {
   private TextView timerView;
   private long baseTimer;
   
   @Override
   protected void onCreate(Bundle savedInstanceState) {
     setContentView(R.layout.chrono);
     MyChronometer.this.baseTimer = SystemClock.elapsedRealtime();
     timerView = (TextView) this.findViewById(R.id.timerView);
     final Handler startTimehandler = new Handler(){
     public void handleMessage(android.os.Message msg) {
         if (null != timerView) {
           timerView.setText((String) msg.obj);
         }
       }
     };
     new Timer("開機計時器").scheduleAtFixedRate(new TimerTask() {
       @Override
       public void run() {
         int time = (int)((SystemClock.elapsedRealtime() - MyChronometer.this.baseTimer) / 1000);
         String hh = new DecimalFormat("00").format(time / 3600);
         String mm = new DecimalFormat("00").format(time % 3600 / 60);
         String ss = new DecimalFormat("00").format(time % 60);  
         String timeFormat = new String(hh + ":" + mm + ":" + ss);
         Message msg = new Message();
         msg.obj = timeFormat;
         startTimehandler.sendMessage(msg);
       }
       
     }, 0, 1000L);
     super.onCreate(savedInstanceState);
   }

新開一個定時器(Timer), 在子線程中獲取開機時間並轉成字符串格式, 利用handler傳回UI線程顯示。

(3)運行結果:

2.調用handler.sendMessagedely(Message msg, long delayMillis)

(1) 布局文件與方法1 相同,運行結果與方法1 相同

(2)Activity文件

public class MyChronometer extends Activity {
   private TextView timerView;
   private long baseTimer;
   
   @Override
   protected void onCreate(Bundle savedInstanceState) {
     setContentView(R.layout.chrono);
     MyChronometer.this.baseTimer = SystemClock.elapsedRealtime();
     timerView = (TextView) this.findViewById(R.id.timerView);
     Handler myhandler = new Handler(){
       public void handleMessage(android.os.Message msg) {
         if (0 == MyChronometer.this.baseTimer) {
           MyChronometer.this.baseTimer = SystemClock.elapsedRealtime();
         }
         
         int time = (int)((SystemClock.elapsedRealtime() - MyChronometer.this.baseTimer) / 1000);
         String hh = new DecimalFormat("00").format(time / 3600);
         String mm = new DecimalFormat("00").format(time % 3600 / 60);
         String ss = new DecimalFormat("00").format(time % 60);  
         if (null != MyChronometer.this.timerView) {
           timerView.setText(hh + ":" + mm + ":" + ss);
         }
         sendMessageDelayed(Message.obtain(this, 0x0), 1000);
       }
     };
     myhandler.sendMessageDelayed(Message.obtain(myhandler, 0x0), 1000);
     super.onCreate(savedInstanceState);
   }

sendMessageDelayed (Message msg, long delayMillis):在 delayMillis/1000 秒後發送消息 msg。

在Handler 的 handleMessage()方法中調用sendMessageDelayed方法, 巧妙的實現了循環。需要注意的是,在Handler外要調用一次startTimehandler.sendMessageDelayed(Message.obtain(startTimehandler, 0x0), 1000);  以作為循環的入口。

3.借助布局Chronometer

(1) 布局文件

 <?xml version="1.0" encoding="utf-8"?>
 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   android:orientation="vertical" >
   
   <Chronometer
     android:id="@+id/chronometer"
     android:layout_width="wrap_content"
     android:layout_height="wrap_content"
     android:layout_gravity="center_horizontal"
     android:textSize="60sp" />
   
</LinearLayout>

布局Chronometer繼承自TextView

(2)Activity文件

 public class MyChronometer extends Activity {
 
   Chronometer chronometer;
   @Override
   protected void onCreate(Bundle savedInstanceState) {
     setContentView(R.layout.chrono);
     chronometer = (Chronometer) this.findViewById(R.id.chronometer);
     chronometer.setBase(SystemClock.elapsedRealtime());
     chronometer.start();
     super.onCreate(savedInstanceState);
   }
 }

 邏輯代碼很簡單,調用chronometer.start()就可以開始計時。

chronometer.setBase(long base):設置起始計時點,這裡設置的是獲取開機時間。

chronometer.start():以上面setBase()設置的時間點為起始點,開始計時,看一下start()的源碼就知道了:

public void start() {
  mStarted = true;
  updateRunning();
 }

調用了updateRunning(), 跟入updateRunning()方法:

   private void updateRunning() {
     boolean running = mVisible && mStarted;
     if (running != mRunning) {
       if (running) {
         updateText(SystemClock.elapsedRealtime());
         dispatchChronometerTick();
         mHandler.sendMessageDelayed(Message.obtain(mHandler, TICK_WHAT), 1000);
       } else {
         mHandler.removeMessages(TICK_WHAT);
       }
       mRunning = running;
     }
   }
   
   private Handler mHandler = new Handler() {
     public void handleMessage(Message m) {
       if (mRunning) {
         updateText(SystemClock.elapsedRealtime());
         dispatchChronometerTick();
         sendMessageDelayed(Message.obtain(this, TICK_WHAT), 1000);
       }
     }
   };

用updateText()方法設置時間顯示。 至於計時循環機制,和方法二相同,同樣是調用了handler的handMessageDelayed方法。

(3)運行結果:

注意:最後說一個關於Chronometer類的常見問題,看到很多人都問用Chronometer類如何設置格式HH:MM:SS的時間。(如果您有此問題請繼續看,沒有問題請忽略)

問這個問題的童鞋先看一下官方文檔的描述:

 If the format string is null, or if you never call setFormat(), the Chronometer will simply display the timer value in "MM:SS" or "H:MM:SS" form.

也就是說默認情況下,使用的格式是"MM:SS" 或者 "H:MM:SS", 然後有童鞋又會問:那到底是"MM:SS" 還是 "H:MM:SS"。我們先看一下源碼:

updateText():

 private synchronized void updateText(long now) {
     long seconds = now - mBase;
     seconds /= 1000;
     String text = DateUtils.formatElapsedTime(mRecycle, seconds);
 
     if (mFormat != null) {
       Locale loc = Locale.getDefault();
       if (mFormatter == null || !loc.equals(mFormatterLocale)) {
         mFormatterLocale = loc;
         mFormatter = new Formatter(mFormatBuilder, loc);
       }
      mFormatBuilder.setLength(0);
       mFormatterArgs[0] = text;
       try {
         mFormatter.format(mFormat, mFormatterArgs);
         text = mFormatBuilder.toString();
       } catch (IllegalFormatException ex) {
         if (!mLogged) {
           Log.w(TAG, "Illegal format string: " + mFormat);
           mLogged = true;
         }
       }
     }    

  setText(text);
   }

調用了DateUtils.formatElapsedTime, 看一下DateUtils.formatElapsedTime裡面都有啥:

 public static String formatElapsedTime(StringBuilder recycle, long elapsedSeconds) {
     Formatter f = new Formatter(sb, Locale.getDefault());
     initFormatStrings();
     if (hours > 0) {
       return f.format(sElapsedFormatHMMSS, hours, minutes, seconds).toString();
     } else {
       return f.format(sElapsedFormatMMSS, minutes, seconds).toString();
     }
   }

代碼較多,我就挑重點截取了,仔細看看上面哪個if(){}else{}語句,你肯定就恍然大悟了吧?

為了我們理論的正確性,將方法三 Activity中的代碼稍作修改:

chronometer.setBase(-18000000);

運行結果如下:

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持本站。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved