- Service在后台运行,不可以与用户直接交互;
- 一个服务不是一个单独的进程。服务对象本身并不意味着它是在自己的进程中运行,除非另有规定,否则它与运行程序是同在一个进程中;
- 一个服务不是一个单独的线程。Service和其他组件一样,默认情况下,Service中的所有代码都是运行在主线程中;
- Service存在的价值虽然不如Activity那么清晰。但是一般都让Service执行耗时较长的操作。例如:播放音乐、下载文件、上传文件等等。但是因为Service默认运行在主线程中,因此不能直接用它来做耗时的请求或者动作,最好在Service中启动新线程来运行耗时的任务;
- 需要通过某一个Activity或其他Context对象来启动Service。context.startService() 或 context.bindService();
- Service很大程度上充当了应用程序后台线程管理器的角色。(如果Activity中新开启一个线程,当该Acitivyt关闭后,该线程依然在工作,但是与开启它的Activity失去联系。也就是说此时的这个线程处于失去管理的状态。但是使用Service,则可以对后台运行的线程有效地管理。)
- 1、Service可以放在独立的进程中,所以更安全;
- 2、使用Service可以依赖现有的binder机制,不需要在应用层面上处理线程同步的繁杂工作;
- 3、系统可以重新启动异常死去的Service。
- 不同点:Activity是与用户交互的组件,即可以看到UI界面,而Service是在后台运行、无需界面;
- 相同点:使用Activity 时我们需要在配置文件中声明<activity>标签,同样的使用Service 也需要在配置文件中声明<service>标签。都具有一定的生命周期。
- Started Service:被启动的服务
-
- 被启动的服务是由其它组件调用startService()方法而启动的,该方法会导致被启动服务的生命周期方法onStartCommand()被回调。当服务是被启动状态后,其生命周期与启动它的组件无关,即使启动服务的组件(Activity,BroadcastReceiver)已经被销毁,该服务还可以在后台无限期运行。除非调用stopSelf()或stopService()来停止该服务。
- Bound Service:被绑定的服务
-
- 绑定服务是允许其它应用程序绑定并且与之交互的Service的实现类。为了提供绑定,必须实现onBind()回调方法。该方法返回IBinder对象,它定义了服务类与Activity交互的程序接口。
- Activity通过bindService()方法绑定到服务类,同时Activity必须提供ServiceConnection接口的实现类,它监视Activity与服务类之间的连接。在重写ServiceConnection接口的onServiceConnected()方法时,实现了将服务类顺利赋值到了Activity中,实现了在Activity中使用该服务类并执行其中的方法。
- START_STICKY(常量值:1):sticky的意思是“粘性的”。使用这个返回值时,我们启动的服务跟应用程序"粘"在一起,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务。当再次启动服务时,传入的第一个参数将为null;
- START_NOT_STICKY(常量值:2):“非粘性的”。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统不会自动重启该服务。
- START_REDELIVER_INTENT(常量值:3):重传Intent。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务,并将Intent的值传入。
- START_STICKY:(常量值:1)车祸后自己苏醒,但是失忆;
- START_NOT_STICKY:(常量值:2)车祸后再也没有苏醒;
- START_REDELIVER_INTENT:(常量值:3)车祸后自己苏醒,依然保持记忆。
(三)、Service的生命周期:
- onCreate():创建服务
- onStartCommand():服务开始运行(在2.0以前版本中,使用onStart()回调方法)
- onDestroy() :服务被停止
- 在程序中调用:context.startService() 会触发执行Service生命周期中的onCreate()、onStartCommand()回调方法,此时服务就开始正式运行;
- 如果Service还没有运行,则android先调用onCreate()然后调用onStartCommand();如果Service已经运行,则只调用onStartCommand(),所以一个Service的onStartCommand方法可能会重复调用多次;
- 如果在程序中调用:context.stopService()会触发执行Service生命周期中的onDestroy()回调方法,会让服务停止;
- stopService()的时候直接onDestroy,如果是调用者自己直接退出而没有调用stopService()的话,Service会一直在后台运行。该Service的调用者再启动该Service后可以通过stopService关闭Service;stopSelf()
- 所以StartService的生命周期为:onCreate --> onStartCommand(可多次调用) --> onDestroy。
- onCreate():创建服务
- onBind():绑定服务,服务开始运行
- onUnbind():取消绑定
- onDestroy() :服务被停止
- 在程序中调用:context.bindService()会触发执行Service生命周期中的onCreate()、onBind()回调方法,此时服务开始运行;
- onBind将返回给客户端一个IBind接口实例,IBind允许客户端回调服务的方法,比如得到Service运行的状态或其他操作。此后调用者(Context,例如Activity)会和Service绑定在一起;
- 如果调用Service的调用者Context退出了,那么会依次调用Service生命周期中的onUnbind()、onDestroy()回调方法,会让服务停止;
- 所以BindService的生命周期为:onCreate --> onBind(只一次,不可多次绑定) --> onUnbind --> onDestory。
- Service是不能自己启动的,只有通过 Context 对象调用startService() 和bindService() 方法来启动。
- 在Service每一次的开启关闭过程中,只有onStartCommand()可被多次调用(通过多次startService调用),其他onCreate()、onBind()、onUnbind()、onDestory()在一个生命周期中只能被调用一次。
- Service可以在和多场合的应用中使用,比如播放多媒体的时候用户启动了其他Activity这个时候程序要在后台继续播放,比如检测SD卡上文件的变化,再或者在后台记录你地理信息位置的改变等等,总之服务总是藏在后头的。
- Activity页面中需要startService(intent) 和 stopService(intent)两个方法来启动Service和停止Service;
- 继承于Service类的自定义子类——MyStartService类中,生命周期回调方法有:onCreate() 、onStartCommand() 、onDestroy();
- 如果停止服务,可以在Activity中调用stopService(intent),也可以intent到Service中执行stopSelf()方法;
- 执行停止服务方法,会回调Service生命周期中的onDestroy()方法;
- 如果希望关闭Activity窗体,服务也停止,那么在Activity的onDestroy()方法中执行stopService()方法。如果希望关闭窗体后,服务还继续,那么Activity的onDestroy()中不执行停止服务即可;
- 在StartService中不会回调onBind()方法;
- 在停止服务后,如果再次点击“播放”,可以重新启动StartSer
- 'vice。
- Creates a default worker thread that executes all intents delivered to onStartCommand() separate from your application's main thread.
- 生成一个默认的且与主线程互相独立的工作者线程来执行所有传送至 onStartCommand() 方法的Intetnt
- Creates a work queue that passes one intent at a time to your onHandleIntent() implementation, so you never have to worry about multi-threading.
- 生成一个工作队列来传送Intent对象给你的onHandleIntent()方法,同一时刻只传送一个Intent对象,这样一来,你就不必担心多线程的问题。
- Stops the service after all start requests have been handled, so you never have to call stopSelf().
- 在所有的请求(Intent)都被执行完以后会自动停止服务,所以,你不需要自己去调用stopSelf()方法来停止该服务
- Provides default implementation of onBind() that returns null.
- 提供了一个onBind()方法的默认实现,它返回null
- Provides a default implementation of onStartCommand() that sends the intent to the work queue and then to your onHandleIntent() implementation
- 提供了一个onStartCommand()方法的默认实现,它将Intent先传送至工作队列,然后从工作队列中每次取出一个传送至onHandleIntent()方法,在该方法中对Intent对相应的处理
@Override
publicint onStartCommand(Intent intent, int flags, int startId) {
imagePath=intent.getStringExtra("imagepath");//获取启动组件传递的图片路径
// 在服务中开启工作线程 下载网络图片 并且存储到sdcard
new Thread(){
publicvoid run() {
if(HttpUtils.isNetWorkConn(DownLoadImageService.this)){
byte[] buff=HttpUtils.getHttpResult(imagePath);
if(buff!=null && buff.length!=0){
if(FileUtils.isStateUseful()){
boolean bl=FileUtils.writeImageToSdcard(buff, imagePath);
if(bl){
System.out.println("写入成功");
stopSelf();//工作完成后停止服务
}else{
System.out.println("写入失败");
}
}else{
System.out.println("sdcard不可用");
}
}else{
System.out.println("下载图片异常");
}
}else{
System.out.println("网络异常");
}
}
}.start();
returnsuper.onStartCommand(intent, flags, startId);
}
- 1、写xml布局文件;
- 2、写MainActivity文件,构建ServiceConnection对象,重写其中的抽象方法onServiceDisconnected()和onServiceConnected();
- 3、写继承于Service的BindService类文件,定义继承于Binder的内部类MyBinder,在其中定义方法getService();
- 4、BindService类中重写onCreate()方法、重写onBind()回调方法,onBind()方法返回MyBinder对象,重写onDestroy()方法;
<RelativeLayoutxmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:paddingBottom="@dimen/activity_vertical_margin"
android:paddingLeft="@dimen/activity_horizontal_margin"
android:paddingRight="@dimen/activity_horizontal_margin"
android:paddingTop="@dimen/activity_vertical_margin"
tools:context=".MainActivity">
<Button
android:id="@+id/button_play"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:onClick="click"
android:text="播放音乐"/>
<Button
android:id="@+id/button_pause"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/button_play"
android:layout_below="@+id/button_play"
android:layout_marginTop="18dp"
android:onClick="click"
android:text="暂停播放"/>
<Button
android:id="@+id/button_stop"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_alignLeft="@+id/button_pause"
android:layout_below="@+id/button_pause"
android:layout_marginTop="20dp"
android:onClick="click"
android:text="停止音乐"/>
</RelativeLayout>
package com.qianfeng.playmusicbinderservice; import com.qianfeng.playmusicbinderservice.PlayMusicService.MyBinder; import android.os.Bundle; import android.os.IBinder; import android.app.Activity; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.ServiceConnection; import android.view.Menu; import android.view.View; public class MainActivity extends Activity { private MyServiceConn serviceConn; private PlayMusicService playMusicService; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); serviceConn=new MyServiceConn(); } public class MyServiceConn implements ServiceConnection{ @Override public void onServiceConnected(ComponentName name, IBinder service) { MyBinder binder=(MyBinder) service; playMusicService=binder.getService(); } @Override public void onServiceDisconnected(ComponentName name) { serviceConn=null; } } public void click(View view){ switch (view.getId()) { case R.id.button_play: if (playMusicService==null) { Intent intent=new Intent(MainActivity.this,PlayMusicService.class); bindService(intent, serviceConn, Context.BIND_AUTO_CREATE); }else{ playMusicService.playMusic(); } break; case R.id.button_pause: if(playMusicService!=null){ playMusicService.pauseMusic(); } break; case R.id.button_stop: if(playMusicService!=null){ playMusicService.stopMusic(); playMusicService=null; unbindService(serviceConn); } break; } } }
package com.qianfeng.playmusicbinderservice; import java.io.IOException; import android.app.Service; import android.content.Intent; import android.media.MediaPlayer; import android.media.MediaPlayer.OnCompletionListener; import android.os.Binder; import android.os.IBinder; public class PlayMusicService extends Service { private MediaPlayer mediaPlayer;//声明操作媒体的对象 private int position;//记录音乐播放的位置 @Override public IBinder onBind(Intent intent) { playMusic();//绑定之后直接播放 return new MyBinder(); } public class MyBinder extends Binder{ public PlayMusicService getService(){ return PlayMusicService.this; } } @Override public void onCreate() { super.onCreate(); if(mediaPlayer==null){ mediaPlayer=mediaPlayer.create(PlayMusicService.this, R.raw.heavencity);//根据操作的媒体资源创建MediaPlayer对象 mediaPlayer.setLooping(false);//设置不需要单曲循环 } //当歌曲播放完毕之后执行的回调接口 mediaPlayer.setOnCompletionListener(new OnCompletionListener() { @Override public void onCompletion(MediaPlayer mp) { mediaPlayer.release();//释放资源 } }); } /* * 播放音乐 */ public void playMusic(){ if(mediaPlayer!=null && !mediaPlayer.isPlaying()){//表示mediaPlayer不为null并且没有播放的歌曲 try { if(position!=0){ mediaPlayer.seekTo(position);//根据指定的下标开始播放 mediaPlayer.start(); }else{ mediaPlayer.stop(); mediaPlayer.prepare();//启动播放之前必须prepare方法 mediaPlayer.start();//启动播放 } } catch (IllegalStateException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } } /* * 暂停音乐 */ public void pauseMusic(){ if(mediaPlayer!=null && mediaPlayer.isPlaying()){ position=mediaPlayer.getCurrentPosition();// 获取当前播放的位置 mediaPlayer.pause();//暂停音乐 } } /* * 停止音乐 */ public void stopMusic(){ if(mediaPlayer!=null){ mediaPlayer.stop();//停止音乐 mediaPlayer.reset();// 资源重置 } } }
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme">
<activity
android:name="com.qianfeng.playmusicbinderservice.MainActivity"
android:label="@string/app_name">
<intent-filter>
<actionandroid:name="android.intent.action.MAIN"/>
<categoryandroid:name="android.intent.category.LAUNCHER"/>
</intent-filter>
</activity>
<serviceandroid:name="com.qianfeng.playmusicbinderservice.PlayMusicService"></service>
</application>
- 与用户正发生交互;
- 它控制一个与用户交互的必须的基本的服务;
- 一个正在调用生命周期回调方法的ervice(如onCreate()、onStar()、onDestroy());
- 一个正在运行onReceive()方法的广播接收对象。