Android Service相关

Android 的 Service 是四大组件之一,有着非常重要的地位。下面来记录一些重要的知识点。

常用方法

方法 说明
startService() 启动服务(两种启动方式之一)
bindService() 绑定服务(两种启动方式之一)
stopService() 关闭服务(对应startService)
unbindService() 解绑服务(对应bindService)
onCreate() 创建服务(生命周期)
onStartCommand() 开始服务(生命周期)
onDestroy() 销毁服务(生命周期)
onBind() 绑定服务(生命周期)
onUnbind() 解绑服务(生命周期)

生命周期

startService

startService -> onCreate -> onStartCommand -> stopService -> onDestory

注意 :多次调用startService,onCreate 只会执行一次,onStartCommand 会多次调用

bindService

bindService -> onCreate -> onBind -> unbindService -> onUnBind -> onDestory

同时使用 startService 和 bindService

startService -> onCreate -> onStartCommand -> bindService -> onBind -> unbindService -> onUnBind -> stopService -> onDestory

特别注意

  • startService()和stopService()只能开启和关闭Service,无法操作Service
  • bindService()和unbindService()可以操作Service
  • startService开启的Service,调用者退出后Service仍然存在
  • bindService开启的Service,调用者退出后,Service随着调用者销毁

Service 的类型及应用场景

Android Service相关_第1张图片
Service类型场景

Service 和 Thread 区别

Service和Thread之间没有任何关系

名称 相同点 不同点
Service 作用:执行异步处理 1、运行在主线程(不能处理耗时操作 否则ANR) 2、依赖进程而非Activity
Thread 作用:执行异步处理 1、运行于工作线程 2、依赖某个Activity

Service 可以和 Thread 配合使用处理耗时操作~

@Override  
public int onStartCommand(Intent intent, int flags, int startId) 
{  
    //新建工作线程
    new Thread(new Runnable() {  
        @Override  
        public void run() {  
            // 开始执行后台任务  
        }  
    }).start();  
    return super.onStartCommand(intent, flags, startId);  
}  

class MyBinder extends Binder {  
    public void service_connect_Activity() {  
      //新建工作线程
        new Thread(new Runnable() {  
            @Override  
            public void run() {  
                // 执行具体的下载任务  
            }  
        }).start();  
    }  
}

Service 和 IntentService 区别

IntentService是Android里面的一个封装类,继承自四大组件之一的Service。用来处理异步请求,实现多线程。(常用于 按顺序、在后台执行 的下载场景)

IntentService源码

public abstract class IntentService extends Service {
private volatile Looper mServiceLooper;
private volatile ServiceHandler mServiceHandler;
private String mName;
private boolean mRedelivery;

private final class ServiceHandler extends Handler {
    public ServiceHandler(Looper looper) {
        super(looper);
    }

    @Override
    public void handleMessage(Message msg) {
        onHandleIntent((Intent)msg.obj);
        stopSelf(msg.arg1);
    }
}
public IntentService(String name) {
    super();
    mName = name;
}
@Override
public void onCreate() {
    // TODO: It would be nice to have an option to hold a partial wakelock
    // during processing, and to have a static startService(Context, Intent)
    // method that would launch the service & hand off a wakelock.

    super.onCreate();
    HandlerThread thread = new HandlerThread("IntentService[" + mName + "]");
    thread.start();

    mServiceLooper = thread.getLooper();
    mServiceHandler = new ServiceHandler(mServiceLooper);
}

@Override
public void onStart(@Nullable Intent intent, int startId) {
    Message msg = mServiceHandler.obtainMessage();
    msg.arg1 = startId;
    msg.obj = intent;
    mServiceHandler.sendMessage(msg);
}
@Override
public int onStartCommand(@Nullable Intent intent, int flags, int startId) {
    onStart(intent, startId);
    return mRedelivery ? START_REDELIVER_INTENT : START_NOT_STICKY;
}

@Override
public void onDestroy() {
    mServiceLooper.quit();
}

/**
 * Unless you provide binding for your service, you don't need to implement this
 * method, because the default implementation returns null.
 * @see android.app.Service#onBind
 */
@Override
@Nullable
public IBinder onBind(Intent intent) {
    return null;
}
@WorkerThread
protected abstract void onHandleIntent(@Nullable Intent intent);

从源码可以看出

  • IntentService本质是采用Handler & HandlerThread
  • 通过HandlerThread单独开启一个名为IntentService的线程
  • 创建一个名叫ServiceHandler的内部Handler
  • 把内部Handler与HandlerThread所对应的子线程进行绑定
  • 通过onStartCommand()传递给服务intent,依次插入到工作队列中,并逐个发送给onHandleIntent()
  • 通过onHandleIntent()来依次处理所有Intent请求对象所对应的任务

因此依据源码注释提示,复写方法onHandleIntent(),再在里面根据Intent的不同进行不同的线程操作就可以了。

注意事项

  • IntentService是以队列执行的,如果一个任务正在IntentService中执行,此时再发送一个新的任务请求,这个新的任务会一直等待直到前面一个任务执行完毕才开始执行。

原因:多次startService时,onCreate只执行一次,只会创建一个工作线程!却会多次执行onStartCommand,由源码可见,只是把消息加入消息队列中等待执行~

  • 不要使用bindService 启动 IntentService

源码中,onBind()是默认返回null的,而采用bindService() 启动 IntentService的生命周期是:onCreate() —>onBind()—>onunbind()—>onDestory()
并不会调用onstart()或者onstartcommand()方法,所以不会将消息发送到消息队列,那么onHandleIntent()将不会回调,即无法实现多线程的操作。

**如上介绍了IntentService,Service 和 IntentService 区别主要是

名称 不同点
Service 1、依赖于应用程序的主线程(不是独立的进程 or 线程)2、需要主动调用stopSelft()来结束服务
IntentService 1、创建一个工作线程来处理多线程任务 2、IntentService不需要(在所有intent被处理完后,系统会自动关闭服务)

你可能感兴趣的:(Android Service相关)