<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <TextView android:layout_width="fill_parent" android:layout_height="wrap_content" android:layout_gravity="center" android:id="@+id/txt" /> <Button android:id="@+id/btnStartTime" android:text="开始计时" android:layout_width="80dip" android:layout_height="wrap_content" ></Button> <Button android:id="@+id/btnStopTime" android:text="中止计时" android:layout_width="80dip" android:layout_height="wrap_content" /> <SeekBar android:id="@+id/SeekBar01" android:layout_width="match_parent" android:layout_height="wrap_content"></SeekBar> </LinearLayout>
这里使用TextView 来显示倒计时的时间变化,两个按钮用于控制时间的开始和中止。SeekBar主要是用于查看线程是否被阻塞(阻塞时没法拖动)。
@Override publicvoid onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); txt = (TextView) findViewById(R.id.txt); btnStart = (Button) findViewById(R.id.btnStartTime); btnStop = (Button) findViewById(R.id.btnStopTime); Log.d("ThreadId", "onCread:" + String.valueOf(Thread.currentThread().getId())); myHandler =new Handler(this); btnStart.setOnClickListener(this); btnStop.setOnClickListener(this); }
发送消息 @Override publicvoid onClick(View v) { switch (v.getId()) { case R.id.btnStartTime: startTimer(); break; case R.id.btnStopTime: timer.cancel(); break; } } privatesynchronizedvoid startTimer() { timer =new Timer(); // TimerTask updateTimerValuesTask = new TimerTask() { // @Override // public void run() { // updateTimerValues(); // } // // }; //自定义的CallBack模式。Task继承自TimerTask Task updateTimerValuesTask =new Task(this); timer.schedule(updateTimerValuesTask, 1000, 1000); } //执行耗时的倒计时任务。 privatevoid updateTimerValues() { total--; Log.d("ThreadId", "send:" + String.valueOf(Thread.currentThread().getId())); Message msg=new Message(); Bundle date =new Bundle();// 存放数据 date.putInt("time", total); msg.setData(date); msg.what=0; myHandler.sendMessage(msg); //另外一种写法 // Message msg=myHandler.obtainMessage(); // Bundle date = new Bundle();// 存放数据 // date.putInt("time", total); // msg.setData(date); // msg.what=0; // msg.sendToTarget(); } @Override publicvoid TaskRun() { updateTimerValues(); }
实现Button按钮的事件处理以此进入倒计时操做。这里使用的Timer 来执行定时操做(其实咱们彻底能够另起一个线程)。Task类继承了TimerTask类,里面增长了一个任务处理接口来实现回调模式,应此Activity须要实现该回调的接口 ITaskCallBack(这样作是由于我比较不喜欢内部类的编写方法)。
ICallBack接口和Task类 publicinterface ITaskCallBack { void TaskRun(); } publicclass Task extends TimerTask { private ITaskCallBack iTask; public Task(ITaskCallBack iTaskCallBack) { super(); iTask=iTaskCallBack; } publicvoid setCallBack(ITaskCallBack iTaskCallBack) { iTask=iTaskCallBack; } @Override publicvoid run() { // TODO Auto-generated method stub iTask.TaskRun(); } }
这是Java的回调函数的通常写法。
实现CallBack /** * 实现消息处理 */ @Override publicboolean handleMessage(Message msg) { switch(msg.what) { case0: Bundle date=msg.getData(); txt.setText(String.valueOf(date.getInt("time"))); Log.d("ThreadId", "HandlerMessage:" + String.valueOf(Thread.currentThread().getId())); Log.d("ThreadId", "msgDate:" + String.valueOf(date.getInt("time"))); break; } returnfalse; }
Activity类 publicclass ThreadHandlerrActivity extends Activity implements Callback, OnClickListener { private TextView txt; private Button btnStart, btnStop; private Handler myHandler; private TimerThread timerThread; privateint Total=30; /** Called when the activity is first created. */ @Override publicvoid onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); txt = (TextView) findViewById(R.id.txt); btnStart = (Button) findViewById(R.id.btnStartTime); btnStop = (Button) findViewById(R.id.btnStopTime); Log.d("ThreadId", "onCread:" + String.valueOf(Thread.currentThread().getId())); myHandler =new Handler(this); btnStart.setOnClickListener(this); btnStop.setOnClickListener(this); } /** * 实现消息处理 */ @Override publicboolean handleMessage(Message msg) { switch(msg.what) { case0: Bundle date=msg.getData(); txt.setText(String.valueOf(date.getInt("time"))); Log.d("ThreadId", "HandlerMessage:" + String.valueOf(Thread.currentThread().getId())); Log.d("ThreadId", "msgDate:" + String.valueOf(date.getInt("time"))); break; } returnfalse; } @Override publicvoid onClick(View v) { switch (v.getId()) { case R.id.btnStartTime: //自定义的线程 timerThread=new TimerThread(myHandler,60); timerThread.start(); break; case R.id.btnStopTime: timerThread.stop(); //timerThread.destroy(); break; } } }
自定义的线程类 ** * 自定义的线程类,经过传入的Handler,和Total 按期执行耗时操做 * @author linzijun * */ publicclass TimerThread extends Thread { publicint Total=60; public Handler handler; /** * 初始化构造函数 * @param mhandler handler 用于发送消息 * @param total 总周期 */ public TimerThread(Handler mhandler,int total) { super(); handler=mhandler; Total=total; } @Override publicvoid run() { while(true) { Total--; if(Total<0) break; try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } Message msg=new Message(); Bundle date =new Bundle();// 存放数据 date.putInt("time", Total); msg.setData(date); msg.what=0; Log.d("ThreadId", "Thread:" + String.valueOf(Thread.currentThread().getId())); handler.sendMessage(msg); } super.run(); } }
这里继承了Thread类,也能够直接实现 Runnable接口。
POST publicclass PostHandler extends Activity implements OnClickListener, Runnable { private TextView txt; private Button btnStart, btnStop; private Handler myHandler; private Timer timer; privateint total =60; @Override protectedvoid onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.main); txt = (TextView) findViewById(R.id.txt); btnStart = (Button) findViewById(R.id.btnStartTime); btnStop = (Button) findViewById(R.id.btnStopTime); Log.d("ThreadId", "onCread:" + String.valueOf(Thread.currentThread().getId())); myHandler =new Handler() { @Override publicvoid handleMessage(Message msg) { switch(msg.what) { case0: Bundle date=msg.getData(); txt.setText(String.valueOf(date.getInt("time"))); Log.d("ThreadId", "HandlerMessage:" + String.valueOf(Thread.currentThread().getId())); Log.d("ThreadId", "msgDate:" + String.valueOf(date.getInt("time"))); break; } } }; btnStart.setOnClickListener(this); btnStop.setOnClickListener(this); } @Override publicvoid onClick(View v) { switch (v.getId()) { case R.id.btnStartTime: //myHandler.post(this); myHandler.postDelayed(this, 1000); break; case R.id.btnStopTime: break; } } @Override publicvoid run() { while(true) { total--; if(total<0) break; try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } Message msg=new Message(); Bundle date =new Bundle();// 存放数据 date.putInt("time", total); msg.setData(date); msg.what=0; Log.d("ThreadId", "POST:" + String.valueOf(Thread.currentThread().getId())); myHandler.sendMessage(msg); Log.d("ThreadId", "Thread:" + String.valueOf(Thread.currentThread().getId())); } } }
使用POST的方式 是将Runnable 一块儿发送给处理的线程(这里为UI),若是Runnable的操做比较耗时的话那线程将进入阻塞状态。能够看到先运行 Runnable的Run方法 而后在进入 HandleMessage() 。我还尝试了另外一种写法,将TimerThreadPOST过去,运行结果是同样的。
POST publicclass PostHandler extends Activity implements OnClickListener, Runnable { private TextView txt; private Button btnStart, btnStop; private Handler myHandler; private Timer timer; privateint total =60; @Override protectedvoid onCreate(Bundle savedInstanceState) { // TODO Auto-generated method stub super.onCreate(savedInstanceState); setContentView(R.layout.main); txt = (TextView) findViewById(R.id.txt); btnStart = (Button) findViewById(R.id.btnStartTime); btnStop = (Button) findViewById(R.id.btnStopTime); Log.d("ThreadId", "onCread:" + String.valueOf(Thread.currentThread().getId())); myHandler =new Handler() { @Override publicvoid handleMessage(Message msg) { switch(msg.what) { case0: Bundle date=msg.getData(); txt.setText(String.valueOf(date.getInt("time"))); Log.d("ThreadId", "HandlerMessage:" + String.valueOf(Thread.currentThread().getId())); Log.d("ThreadId", "msgDate:" + String.valueOf(date.getInt("time"))); break; } } }; btnStart.setOnClickListener(this); btnStop.setOnClickListener(this); } @Override publicvoid onClick(View v) { switch (v.getId()) { case R.id.btnStartTime: //myHandler.post(this); myHandler.postDelayed(this, 1000); break; case R.id.btnStopTime: break; } } @Override publicvoid run() { while(true) { total--; if(total<0) break; try { Thread.sleep(1000); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } Message msg=new Message(); Bundle date =new Bundle();// 存放数据 date.putInt("time", total); msg.setData(date); msg.what=0; Log.d("ThreadId", "POST:" + String.valueOf(Thread.currentThread().getId())); myHandler.sendMessage(msg); Log.d("ThreadId", "Thread:" + String.valueOf(Thread.currentThread().getId())); } } }
能够说POST的各类方法主要是用于 “按计划发送消息或执行某个Runnanble(使用POST方法)”。