在做Service简单练习时,在Service中的OnCreate、OnStart、OnDestroy三个方法中都像在Activity中同样的方法调用了Toast.makeText,并在Acitivy中通过两个按钮来调用该服务的onStart和onDestroy方法:
DemoService代码如下:
@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();
}
//运行之后,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.”
//果然还是Context的问题,所以想要toast能够正常工作,需要在Activity的主线程上运行才行,那么如何得到主线程UI的Context呢?可以通过Handler将一个自定义的线程运行于主线程之上。
再来看一下Toast.show方法的src:
public void show() {
...
service.enqueueToast(pkg, tn, mDuration); //将该toast插入到一个消息队列中
...
}
原理上看,Android中大致上是消息队列和消息循环,主线程从消息队列中取得消息并处理。而Handler看作是一个工具类,用来向消息队列中插入消息。所以我们重构原来的代码:这样就可以显示toast。
//方法一:在service上使用消息队列和消息循环
@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();
}
});
}
onStart() onDestory()同上。
//方法二:引用主线程的context,MainActivity.context。如下所示:
//主线程代码
public class MainActivity extends Activity {
/** Called when the activity is first created. */
public static Context mainContext;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mainContext = MainActivity.this;
}
//service代码
@Override
public void onCreate(){
super.onCreate();
Toast.makeText(MainActivity.mainContext, "Service is created!", Toast.LENGTH_LONG).show();
}
//如上代码所示,只要在service上引用主线程的context,就能在service组件上显示toast插件。
//总结:在Android的Framework中使用Toast,要将Toast添加到主线程里才能正常工作。