大略地看了一下android.app下的Service类,觉得它与Activity非常相似,只是要注意几个地方:
1.生命周期,Service的从onCreate()->onStart(int,Bundle)->onDestroy()显得更为简单。但是它的onStart是带参数的,第一个ID可用来标识这个service,第二个参数显示是用来传递数据的了。比较Activity,传递数据的Bundle是在onCreate就带进入的。
2.Service的启动由Context.startService开始,其实Activity或者Service都是Context的派生类。结束于Context.stopService()或者它自己的stopSelf()。
3.Service还有一个与Activity不一样的是它可以由另一个Context去绑定一个已存在的Service。就是这个方法Context.bindService(),被绑定的Service要求是已经onCreate了但可以没有onStart。在Service类中有个抽象方法getBinder()可以得到这个IBinder对象。关于这方面的细节,以后再看,这里只做个记录罢。
4.与Service有关的还有一个安全的问题,可以在AndroidManifest.xml中用<uses-permission>标签来声明一个Service的访问权限,关于Android的安全问题也留待以后再解决吧。
我一直相信一种水到渠成的学习方法,先从最简单的东西入手,就不会觉得学习很枯燥了。
下面来做个例子。
修改AndroidManifest.xml文件,增加一个Activity和一个Service:
<
activity
class
=".HelloTwoD"
android:label
="hello_two_d"
>
</
activity
>
<
service
class
=".HelloTwoDService"
/>
HelloTwoD.java的代码比较简单,如下:
public
class
HelloTwoD
extends
Activity
implements
OnClickListener
...
{
publicHelloTwoD()
...{
super();
}
publicvoidonCreate(Bundleicicle)...{
super.onCreate(icicle);
setTheme(android.R.style.Theme_Dark);
setContentView(R.layout.maind);
Buttonbtn=(Button)findViewById(R.id.btnTest);
btn.setOnClickListener(this);
}
@Override
publicvoidonClick(Viewarg0)...{
//用一个显式的Intent来启动服务
Intenti=newIntent();
i.setClass(this,HelloTwoDService.class);
//带上我的名字
Bundleb=newBundle();
b.putString("name","sharetop");
this.startService(i,b);
}
}
当然要启动这个HelloTwoD,也需要在我最初的那个HelloTwo中加一点代码(我就不罗嗦了)。再看看那个HelloTwoDService是如何实现的:
public
class
HelloTwoDService
extends
Service
...
{
publicTimertimer;
publicfinalStringTAG="HelloTwoDService_TAG";
publicvoidonCreate()...{
super.onCreate();
Log.d(TAG,"onCreate");
timer=newTimer(true);
}
@Override
publicIBindergetBinder()...{
//TODOAuto-generatedmethodstub
returnnull;
}
publicvoidonStart(intstartId,Bundlearg)
...{
//看看startId是什么内容
if(arg!=null)
Log.d(TAG,"onStart"+Integer.valueOf(startId).toString()+"from"+arg.getString("name"));
else
Log.d(TAG,"onStartwithnullBundle");
timer.schedule(newTimerTask()...{
publicvoidrun()...{
//表示一下我的存在
Log.d(TAG,"sayfromatimer.");
//停掉自己这个服务
HelloTwoDService.this.stopSelf();
}
},5000);
}
publicvoidonDestroy()
...{
Log.d(TAG,"onDestroy");
}
}
这里我用一个定时器timer来延时5秒钟显示消息,否则立即就显示出来觉得不象一个后台服务了。用日志输出那个onStart中的startId看看,原来只是一个标识而已。
下面来个简单的NotificationManager吧,看了看API文档,觉得最简单地恐怕就是那个NotificationManager.notifyWithText()了,修改上面的run方法如下:
timer.schedule(
new
TimerTask()
...
{
publicvoidrun()...{
NotificationManagermanager=(NotificationManager)getSystemService(NOTIFICATION_SERVICE);
manager.notifyWithText(1001,"わたしはSHARETOPです。",NotificationManager.LENGTH_LONG,null);
HelloTwoDService.this.stopSelf();
}
}
,
5000
);
再试试看效果。太简单了,Notification主要是用于后台服务用来通知前台,所以,Android提供了三类不同的通知方式,notifyWithText可以简单地显示一个字串,而notifyWithView稍复杂点,可以有一个view来构造这个显示信息框,而最灵活的就是那个notify(int id, Notification notification)了,参数notification是类Notification的实例。
修改一下刚才的那个run方法,如下:
timer.schedule(
new
TimerTask()
...
{
publicvoidrun()...{
NotificationManagermanager=(NotificationManager)getSystemService(NOTIFICATION_SERVICE);
Notificationnf=newNotification(R.drawable.icon,"这是信息的详细描述",null,"信息的标题",null);
manager.notify(0,nf);
HelloTwoDService.this.stopSelf();
}
}
,
5000
);
这里创建一个Notification的实例nf,构造函数的第一个参数是那个显示在状态栏(也就是Android手机上面的那一条显示信号强度、电池电量等信息的位置)的图标。后面可以有
一个标题和点击以后的详细信息,这是字串形式,还可以有一个Intent用来表示点击后可以发生一个跳转行为。
OK,今天到此为止。下次该体验各种View了。