Service的使用(二)启动服务与IntentService

来源:互联网 发布:php unset函数 编辑:程序博客网 时间:2024/05/21 11:07

创建Started Srevice

public class StartedService extends Service {    private static final String TAG = "StartedService";    public StartedService() {    }    @Override    public void onCreate() {        super.onCreate();        Log.d(TAG, "onCreate: ");    }    @Override    public int onStartCommand(Intent intent, int flags, int startId) {        Log.d(TAG, "onStartCommand: ");        return super.onStartCommand(intent, flags, startId);    }    @Override    public IBinder onBind(Intent intent) {        return null;    }    @Override    public void onDestroy() {        super.onDestroy();        Log.d(TAG, "onDestroy: ");    }}

服务启动之后,其生命周期即独立于启动它的组件,并且可以在后台无限期地运行,即使启动服务的组件已被销毁也不受影响。 因此,服务应通过调用 stopSelf() 结束工作来自行停止运行,或者由另一个组件通过调用 stopService() 来停止它

Intent intent = new Intent(ServiceActivity.this, StartedService.class);//开始服务startService(intent);//停止服务stopService(intent)

首次启动Service时,onCreate才会执行。每次启动Service时,onStartCommand都会执行。如果启动时已经存在服务的实例,onCreate不会执行。

IntentService (异步的,处理完请求会自行停止)

所有的请求会进入工作队列,然后使用工作线程逐一处理所有启动请求。 只需实现 onHandleIntent() 方法即可,该方法会接收每个启动请求的 Intent。

IntentService 的属性:

  1. 创建默认的工作线程,用于在应用的主线程外执行传递给 onStartCommand() 的所有 Intent。
  2. 创建工作队列,用于将一个 Intent 逐一传递给 onHandleIntent() 实现,这样就不必担心多线程问题。
  3. 在处理完所有启动请求后停止服务,不必调用 stopSelf()。
  4. 提供 onBind() 的默认实现(返回 null)。
  5. 提供 onStartCommand() 的默认实现,可将 Intent 依次发送到工作队列和 onHandleIntent() 实现。

综上,只需实现 onHandleIntent() 来完成客户端提供的工作即可。(不过还需要为服务提供小型构造函数。)

示例

public class HelloIntentService extends IntentService {  public HelloIntentService() {      super("HelloIntentService");  }  /**   * The IntentService calls this method from the default worker thread with   * the intent that started the service. When this method returns, IntentService   * stops the service, as appropriate.   */  //只需实现onHandIntent即可  @Override  protected void onHandleIntent(Intent intent) {      // Normally we would do some work here, like download a file.      // For our sample, we just sleep for 5 seconds.      long endTime = System.currentTimeMillis() + 5*1000;      while (System.currentTimeMillis() < endTime) {          synchronized (this) {              try {                  wait(endTime - System.currentTimeMillis());              } catch (Exception e) {              }          }      }  }}

若有必要重写其他回调方法(如 onCreate()、onStartCommand() 或 onDestroy()),要确保调用超类实现,以便 IntentService 能够妥善处理工作线程的生命周期
例如:

@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {    Toast.makeText(this, "service starting", Toast.LENGTH_SHORT).show();    //调用超类    return super.onStartCommand(intent,flags,startId);}

多线程处理请求

public class HelloService extends Service {  private Looper mServiceLooper;  private ServiceHandler mServiceHandler;  // Handler that receives messages from the thread  private final class ServiceHandler extends Handler {      public ServiceHandler(Looper looper) {          super(looper);      }      @Override      public void handleMessage(Message msg) {          // Normally we would do some work here, like download a file.          // For our sample, we just sleep for 5 seconds.          long endTime = System.currentTimeMillis() + 5*1000;          while (System.currentTimeMillis() < endTime) {              synchronized (this) {                  try {                      wait(endTime - System.currentTimeMillis());                  } catch (Exception e) {                  }              }          }          // Stop the service using the startId, so that we don't stop          // the service in the middle of handling another job          stopSelf(msg.arg1);      }  }  @Override  public void onCreate() {    // Start up the thread running the service.  Note that we create a    // separate thread because the service normally runs in the process's    // main thread, which we don't want to block.  We also make it    // background priority so CPU-intensive work will not disrupt our UI.    HandlerThread thread = new HandlerThread("ServiceStartArguments",            Process.THREAD_PRIORITY_BACKGROUND);    thread.start();    // Get the HandlerThread's Looper and use it for our Handler    mServiceLooper = thread.getLooper();    mServiceHandler = new ServiceHandler(mServiceLooper);  }  @Override  public int onStartCommand(Intent intent, int flags, int startId) {      Toast.makeText(this, "service starting", Toast.LENGTH_SHORT).show();      // For each start request, send a message to start a job and deliver the      // start ID so we know which request we're stopping when we finish the job      Message msg = mServiceHandler.obtainMessage();      msg.arg1 = startId;      mServiceHandler.sendMessage(msg);      // If we get killed, after returning from here, restart      return START_STICKY;  }  @Override  public IBinder onBind(Intent intent) {      // We don't provide binding, so return null      return null;  }  @Override  public void onDestroy() {    Toast.makeText(this, "service done", Toast.LENGTH_SHORT).show();  }}

onStartCommand的返回值

onStartCommand() 方法必须返回整型数,用于描述系统应该如何在服务终止的情况下继续运行服务。
必须是以下值之一:

START_NOT_STICKY
如果系统在 onStartCommand() 返回后终止服务,则除非有挂起 Intent 要传递,否则系统不会重建服务。这是最安全的选项,可以避免在不必要时以及应用能够轻松重启所有未完成的作业时运行服务。

START_STICKY
如果系统在 onStartCommand() 返回后终止服务,则会重建服务并调用 onStartCommand(),但绝对不会重新传递最后一个 Intent。相反,除非有挂起 Intent 要启动服务(在这种情况下,将传递这些 Intent ),否则系统会通过空 Intent 调用 onStartCommand()。这适用于不执行命令、但无限期运行并等待作业的媒体播放器(或类似服务)。

START_REDELIVER_INTENT
如果系统在 onStartCommand() 返回后终止服务,则会重建服务,并通过传递给服务的最后一个 Intent 调用 onStartCommand()。任何挂起 Intent 均依次传递。这适用于主动执行应该立即恢复的作业(例如下载文件)的服务。

与IntentService的区别

以上示例中,每个请求都分配一个工作线程来处理请求,且每个工作线程执行完任务后就会关闭,不会处理其他请求。而IntentService自始至终只有一个工作线程,处理完一个请求后,继续处理队列中的请求,不会关闭。另外,虽然都是处理很多请求,以上示例其实是同时处理,而IntentService是逐一处理。(实际应用中貌似都用IntentService)

0 0
原创粉丝点击