作者:陈旭
在做Service简单练习时,在Service中的OnCreate、OnStart、OnDestroy 三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法:
DemoService代码如下:
<span style= "font-size:16px;" > @Override public void onCreate() { super .onCreate(); Toast.makeText(getApplicationContext(), "Service is created!" , Toast.LENGTH_LONG).show(); } @Override public void onStart(Intent intent, int startId) { super .onStart(intent, startId); Toast.makeText(getApplicationContext(), "Service is on!" , Toast.LENGTH_LONG).show(); } @Override public void onDestroy(){ super .onDestroy(); Toast.makeText(getApplicationContext(), "Service is off!" , Toast.LENGTH_LONG).show(); } </span>
运行之后,DemoService中的信息都没有显示出来。
刚开始以为所得到的Context不正确,在Service直接调用getApplicationContext()得到的是Service的Context,但是细究来看,Toast应该得到主UI的Context才能显示,所以找了一下,Google对Toast的说明中,有一句:
“A toast can be created and displayed from an Activity or Service . If you create a toast notification from a Service,it appears in front of the Activity currently in focus. ”
(http://developer.Android .com/guide/topics/ui/notifiers/toasts.html)
那么按照这句来看,service中创建的toast会在Acivity的UI前面聚焦显示。但为什么运行没有效果呢?再来查看一下makeText方法。
果然还是Context的问题,所以想要toast能够正常工作,需要在Activity的主线程上运行才行,那么如何得到主线程UI的Context呢?可以通过Handler将一个自定义的线程运行于主线程之上。
再来看一下Toast.show方法的src:
<span style= "font-size:16px;" > public void show() { ... service.enqueueToast(pkg, tn, mDuration); ... } </span>
原理上看,android 中大致上是消息队列和消息循环,主线程从消息队列中取得消息并处理。而Handler看作是一个工具类,用来向消息队列中插入消息。所以我们重构原来的代码:
<span style= "font-size:16px;" > @Override public void onCreate() { super .onCreate(); handler=new Handler(Looper.getMainLooper()); handler.post(new Runnable(){ public void run(){ Toast.makeText(getApplicationContext(), "Service is created!" , Toast.LENGTH_LONG).show(); } }); } @Override public void onStart(Intent intent, int startId) { super .onStart(intent, startId); handler=new Handler(Looper.getMainLooper()); handler.post(new Runnable(){ public void run(){ Toast.makeText(getApplicationContext(), "Service is on!" , Toast.LENGTH_LONG).show(); } }); } @Override public void onDestroy(){ super .onDestroy(); handler=new Handler(Looper.getMainLooper()); handler.post(new Runnable(){ public void run(){ Toast.makeText(getApplicationContext(), "Service is off!" , Toast.LENGTH_LONG).show(); } }); </span>
运行之后的效果如下:
总结:在Android的Framework中使用Toast,要将Toast添加到主线程里才能正常工作