Service中使用Toast显示问题

作者:陈旭 

在做Service简单练习时,在Service中的OnCreate、OnStart、OnDestroy三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法:

DemoService代码如下:


[java]  view plain  copy
  1. "font-size:16px;">@Override  
  2.     public void onCreate()  
  3.     {  
  4.         super.onCreate();         
  5.         Toast.makeText(getApplicationContext(), "Service is created!", Toast.LENGTH_LONG).show();  
  6.     }  
  7.     @Override  
  8.     public void onStart(Intent intent,int startId)  
  9.     {  
  10.         super.onStart(intent, startId);  
  11.         Toast.makeText(getApplicationContext(), "Service is on!", Toast.LENGTH_LONG).show();  
  12.     }  
  13.     @Override  
  14.     public void onDestroy(){  
  15.         super.onDestroy();  
  16.         Toast.makeText(getApplicationContext(), "Service is off!", Toast.LENGTH_LONG).show();  
  17. }  
  18.   

运行之后,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:

[java]  view plain  copy
  1. "font-size:16px;">public void show() {  
  2.        ...  
  3.         service.enqueueToast(pkg, tn, mDuration);   //将该toast插入到一个消息队列中  
  4.         ...  
  5.     }  
  6.   

原理上看,android中大致上是消息队列和消息循环,主线程从消息队列中取得消息并处理。而Handler看作是一个工具类,用来向消息队列中插入消息。所以我们重构原来的代码:

[java]  view plain  copy
  1. "font-size:16px;">@Override  
  2.     public void onCreate()  
  3.     {  
  4.         super.onCreate();  
  5.         handler=new Handler(Looper.getMainLooper());  
  6.         handler.post(new Runnable(){  
  7.             public void run(){  
  8.                 Toast.makeText(getApplicationContext(), "Service is created!", Toast.LENGTH_LONG).show();  
  9.             }  
  10.         });  
  11.     }  
  12.     @Override  
  13.     public void onStart(Intent intent,int startId)  
  14.     {  
  15.         super.onStart(intent, startId);  
  16.         handler=new Handler(Looper.getMainLooper());  
  17.         handler.post(new Runnable(){  
  18.             public void run(){  
  19.                 Toast.makeText(getApplicationContext(), "Service is on!", Toast.LENGTH_LONG).show();  
  20.             }  
  21.         });  
  22.     }  
  23.     @Override  
  24.     public void onDestroy(){  
  25.         super.onDestroy();  
  26.         handler=new Handler(Looper.getMainLooper());  
  27.         handler.post(new Runnable(){  
  28.             public void run(){  
  29.                 Toast.makeText(getApplicationContext(), "Service is off!", Toast.LENGTH_LONG).show();  
  30.             }  
  31.         });  
  32.   

运行之后的效果如下:




总结:在Android的Framework中使用Toast,要将Toast添加到主线程里才能正常工作

你可能感兴趣的:(Android,基础)