startService与bindService的区别

Android执行Service有两种方法,一种是startService,一种是bindService。下面让咱们一块儿来聊一聊这两种执行Service方法的区别。java

一、生命周期上的区别android

    执行startService时,Service会经历onCreate->onStartCommand。当执行stopService时,直接调用onDestroy方法。调用者若是没有stopService,Service会一直在后台运行,下次调用者再起来仍然能够stopService。
架构

    执行bindService时,Service会经历onCreate->onBind。这个时候调用者和Service绑定在一块儿。调用者调用unbindService方法或者调用者Context不存在了(如Activity被finish了),Service就会调用onUnbind->onDestroy。这里所谓的绑定在一块儿就是说二者共存亡了。ide

    屡次调用startService,该Service只能被建立一次,即该Service的onCreate方法只会被调用一次。可是每次调用startService,onStartCommand方法都会被调用。Service的onStart方法在API 5时被废弃,替代它的是onStartCommand方法。性能

    第一次执行bindService时,onCreate和onBind方法会被调用,可是屡次执行bindService时,onCreate和onBind方法并不会被屡次调用,即并不会屡次建立服务和绑定服务。
this

二、调用者如何获取绑定后的Service的方法spa

    onBind回调方法将返回给客户端一个IBinder接口实例,IBinder容许客户端回调服务的方法,好比获得Service运行的状态或其余操做。咱们须要IBinder对象返回具体的Service对象才能操做,因此说具体的Service对象必须首先实现Binder对象。
code

三、既使用startService又使用bindService的状况xml

    若是一个Service又被启动又被绑定,则该Service会一直在后台运行。首先无论如何调用,onCreate始终只会调用一次。对应startService调用多少次,Service的onStart方法便会调用多少次。Service的终止,须要unbindService和stopService同时调用才行。无论startService与bindService的调用顺序,若是先调用unbindService,此时服务不会自动终止,再调用stopService以后,服务才会终止;若是先调用stopService,此时服务也不会终止,而再调用unbindService或者以前调用bindService的Context不存在了(如Activity被finish的时候)以后,服务才会自动中止。
对象

    那么,什么状况下既使用startService,又使用bindService呢?

    若是你只是想要启动一个后台服务长期进行某项任务,那么使用startService即可以了。若是你还想要与正在运行的Service取得联系,那么有两种方法:一种是使用broadcast,另外一种是使用bindService。前者的缺点是若是交流较为频繁,容易形成性能上的问题,然后者则没有这些问题。所以,这种状况就须要startService和bindService一块儿使用了。

    另外,若是你的服务只是公开一个远程接口,供链接上的客户端(Android的Service是C/S架构)远程调用执行方法,这个时候你能够不让服务一开始就运行,而只是bindService,这样在第一次bindService的时候才会建立服务的实例运行它,这会节约不少系统资源,特别是若是你的服务是远程服务,那么效果会越明显(固然在Servcie建立的是偶会花去必定时间,这点须要注意)。    

四、本地服务与远程服务

    本地服务依附在主进程上,在必定程度上节约了资源。本地服务由于是在同一进程,所以不须要IPC,也不须要AIDL。相应bindService会方便不少。缺点是主进程被kill后,服务变会终止。

    远程服务是独立的进程,对应进程名格式为所在包名加上你指定的android:process字符串。因为是独立的进程,所以在Activity所在进程被kill的是偶,该服务依然在运行。缺点是该服务是独立的进程,会占用必定资源,而且使用AIDL进行IPC稍微麻烦一点。

    对于startService来讲,不论是本地服务仍是远程服务,咱们须要作的工做都同样简单。

五、代码实例

    startService启动服务

public class LocalService1 extends Service {
    /**
    * onBind 是 Service 的虚方法,所以咱们不得不实现它。
    * 返回 null,表示客服端不能创建到此服务的链接。
    */
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    
    @Override
    public void onCreate() {
        super.onCreate();
    }
    
    @Override
    public void onStartCommand(Intent intent, int startId, int flags) {
        super.onStartCommand(intent, startId, flags);
    }
    
    @Override
    public void onDestroy() {
        super.onDestroy();
    }
}

    bindService绑定服务

public class LocalService extends Service {
/**
* 在 Local Service 中咱们直接继承 Binder 而不是 IBinder,由于 Binder 实现了 IBinder 接口,这样咱们能够** 少作不少工做。
*/
public class SimpleBinder extends Binder{
/**
* 获取 Service 实例
* @return
*/
public LocalService getService(){
return LocalService.this;
}

public int add(int a, int b){
return a + b;
}
}

public SimpleBinder sBinder;

@Override
public void onCreate() {
super.onCreate();
// 建立 SimpleBinder
sBinder = new SimpleBinder();
}

@Override
public IBinder onBind(Intent intent) {
// 返回 SimpleBinder 对象
return sBinder;
}
}

     上面的代码关键之处,在于 onBind(Intent) 这个方法 返回了一个实现了 IBinder 接口的对象,这个对象将用于绑定Service 的 Activity 与 Local Service 通讯。

    下面是 Activity 中的代码:

public class Main extends Activity {
    private final static String TAG = "SERVICE_TEST";
    private ServiceConnection sc;
    private boolean isBind;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        sc = new ServiceConnection() {
            @Override
            public void onServiceDisconnected(ComponentName name) {
            }
    
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                LocalService.SimpleBinder sBinder = (LocalService.SimpleBinder)service;
                Log.v(TAG, "3 + 5 = " + sBinder.add(3, 5));
                Log.v(TAG, sBinder.getService().toString());
            }
        };

        findViewById(R.id.btnBind).setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                bindService(new Intent(Main.this, LocalService.class), sc, Context.BIND_AUTO_CREATE);
                isBind = true;
            }
        });

        findViewById(R.id.btnUnbind).setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                if(isBind){
                    unbindService(sc);
                    isBind = false;
                }
            }
        });
    }
}


六、在AndroidManifest.xml里Service元素常见选项  

    android:name  -------------  服务类名

    android:label  --------------  服务的名字,若是此项不设置,那么默认显示的服务名则为类名

    android:icon  --------------  服务的图标

    android:permission  -------  申明此服务的权限,这意味着只有提供了该权限的应用才能控制或链接此服务

    android:process  ----------  表示该服务是否运行在另一个进程,若是设置了此项,那么将会在包名后面加上这段字符串表示另外一进程的名字

    android:enabled  ----------  表示是否能被系统实例化,为true表示能够,为false表示不能够,默认为true

    android:exported  ---------  表示该服务是否可以被其余应用程序所控制或链接,不设置默认此项为 false

相关文章
相关标签/搜索