android service
1. 概念
开始,先稍稍讲一点Android中Service的概念和用途吧~
Service分为本地服务(localService)和远程服务(RemoteService):
1、本地服务依附在主进程上而不是独立的进程,这样在一定程度上节约了资源,另外Local服务因为是在同一进程因此不需要IPC,
也不需要aiDL。相应bindservice会方便很多。主进程被Kill后,服务便会终止。
2、远程服务为独立的进程,对应进程名格式为所在包名加上你指定的android:process字符串。由于是独立的进程,因此在Activity所在进程被Kill的时候,该服务依然在运行,
不受其他进程影响,有利于为多个进程提供服务具有较高的灵活性。该服务是独立的进程,会占用一定资源,并且使用AIDL进行IPC稍微麻烦一点。
按使用方式可以分为以下三种:
1、startService 启动的服务:主要用于启动一个服务执行后台任务,不进行通信。停止服务使用stopService;
2、bindService 启动的服务:该方法启动的服务可以进行通信。停止服务使用unbindService;
3、startService 同时也 bindService 启动的服务:停止服务应同时使用stepService与unbindService
2. Service 与 Thread 的区别
很多时候,你可能会问,为什么要用 Service,而不用 Thread 呢,因为用 Thread 是很方便的,比起 Service 也方便多了,下面我详细的来解释一下。
1). Thread:Thread 是程序执行的最小单元,它是分配cpu的基本单位。可以用 Thread 来执行一些异步的操作。
2). Service:Service 是android的一种机制,当它运行的时候如果是Local Service,那么对应的 Service 是运行在主进程的 main 线程上的。如:onCreate,onStart 这些函数在被系统调用的时候都是在主进程的 main 线程上运行的。如果是RemoteService,那么对应的 Service 则是运行在独立进程的 main 线程上。因此请不要把 Service 理解成线程,它跟线程半毛钱的关系都没有!
既然这样,那么我们为什么要用 Service 呢?其实这跟 android 的系统机制有关,我们先拿 Thread 来说。Thread 的运行是独立于 Activity 的,也就是说当一个 Activity 被 finish 之后,如果你没有主动停止 Thread 或者 Thread 里的 run 方法没有执行完毕的话,Thread 也会一直执行。因此这里会出现一个问题:当 Activity 被 finish 之后,你不再持有该 Thread 的引用。另一方面,你没有办法在不同的 Activity 中对同一 Thread 进行控制。
举个例子:如果你的 Thread 需要不停地隔一段时间就要连接服务器做某种同步的话,该 Thread 需要在 Activity 没有start的时候也在运行。这个时候当你 start 一个 Activity 就没有办法在该 Activity 里面控制之前创建的 Thread。因此你便需要创建并启动一个 Service ,在 Service 里面创建、运行并控制该 Thread,这样便解决了该问题(因为任何 Activity 都可以控制同一 Service,而系统也只会创建一个对应 Service 的实例)。
因此你可以把 Service 想象成一种消息服务,而你可以在任何有 context 的地方调用 Context.startService、Context.stopService、Context.bindService,Context.unbindService,来控制它,你也可以在 Service 里注册 BroadcastReceiver,在其他地方通过发送 broadcast 来控制它,当然这些都是 Thread 做不到的。
3. Service和Activity通信需要用到bindService,通过onBind()方法来实现,看下面bindService的例子
4. Service的生命周期
Service生命周期.png
第一种方式:通过StartService启动Service
通过startService启动后,service会一直无限期运行下去,只有外部调用了stopService()或stopSelf()方法时,该Service才会停止运行并销毁。
要创建一个这样的Service,你需要让该类继承Service类,然后重写以下方法:
onCreate()
1.如果service没被创建过,调用startService()后会执行onCreate()回调;
2.如果service已处于运行中,调用startService()不会执行onCreate()方法。
也就是说,onCreate()只会在第一次创建service时候调用,多次执行startService()不会重复调用onCreate(),此方法适合完成一些初始化工作。
onStartcommand()
如果多次执行了Context的startService()方法,那么Service的onStartCommand()方法也会相应的多次调用。onStartCommand()方法很重要,我们在该方法中根据传入的intent参数进行实际的操作,比如会在此处创建一个线程用于下载数据或播放音乐等。
onBind()
Service中的onBind()方法是抽象方法,Service类本身就是抽象类,所以onBind()方法是必须重写的,即使我们用不到。
onDestory()
在销毁的时候会执行Service该方法。
这几个方法都是回调方法,且在主线程中执行,由android操作系统在合适的时机调用。
startService代码实例
创建TestOneService,并在manifest里注册。
需要注意,项目中的每一个Service都必须在Androidmanifest.xml中注册才行,所以还需要编辑AndroidManifest.xml文件,代码如下所示:
[html] view plain copy
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.example.servicetest"
- android:versionCode="1"
- android:versionName="1.0" >
- <uses-sdk
- android:minSdkVersion="14"
- android:targetSdkVersion="17" />
- <APPlication
- android:allowBackup="true"
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name"
- android:theme="@style/AppTheme" >
- ……
- <service android:name="com.example.servicetest.TestOneService" >
- </service>
- </application>
- </manifest>
在MainActivty中操作TestOneService,code如下:
[java] view plain copy
- /**
- * Created by Kathy on 17-2-6.
- */
- public class TestOneService extends Service{
- @Override
- public void onCreate() {
- Log.i("Kathy","onCreate - Thread ID = " + Thread.currentThread().getId());
- super.onCreate();
- }
- @Override
- public int onStartCommand(Intent intent, int flags, int startId) {
- Log.i("Kathy", "onStartCommand - startId = " + startId + ", Thread ID = " + Thread.currentThread().getId());
- return super.onStartCommand(intent, flags, startId);
- }
- @Nullable
- @Override
- public IBinder onBind(Intent intent) {
- Log.i("Kathy", "onBind - Thread ID = " + Thread.currentThread().getId());
- return null;
- }
- @Override
- public void onDestroy() {
- Log.i("Kathy", "onDestroy - Thread ID = " + Thread.currentThread().getId());
- super.onDestroy();
- }
- }
在MainActivity中三次startService,之后stopService。
[java] view plain copy
- /**
- * Created by Kathy on 17-2-6.
- */
- public class MainActivity extends AppCompatActivity {
- @Override
- protected void onCreate(Bundle savedinstanceState) {
- super.onCreate(savedInstanceState);
- setcontentView(R.layout.activity_main);
- Log.i("Kathy", "Thread ID = " + Thread.currentThread().getId());
- Log.i("Kathy", "before StartService");
- //连续启动Service
- Intent intentOne = new Intent(this, TestOneService.class);
- startService(intentOne);
- Intent intentTwo = new Intent(this, TestOneService.class);
- startService(intentTwo);
- Intent intentThree = new Intent(this, TestOneService.class);
- startService(intentThree);
- //停止Service
- Intent intentFour = new Intent(this, TestOneService.class);
- stopService(intentFour);
- //再次启动Service
- Intent intentFive = new Intent(this, TestOneService.class);
- startService(intentFive);
- Log.i("Kathy", "after StartService");
- }
- }
打印出的Log如下:
[java] view plain copy
- 02-06 15:19:45.090 8938-8938/? I/Kathy: Thread ID = 1
- 02-06 15:19:45.090 8938-8938/? I/Kathy: before StartService
- 02-06 15:19:45.233 8938-8938/? I/Kathy: onCreate - Thread ID = 1
- 02-06 15:19:45.234 8938-8938/? I/Kathy: onStartCommand - startId = 1, Thread ID = 1
- 02-06 15:19:45.234 8938-8938/? I/Kathy: onStartCommand - startId = 2, Thread ID = 1
- 02-06 15:19:45.235 8938-8938/? I/Kathy: onStartCommand - startId = 3, Thread ID = 1
- 02-06 15:19:45.236 8938-8938/? I/Kathy: onDestroy - Thread ID = 1
- 02-06 15:19:45.237 8938-8938/? I/Kathy: onCreate - Thread ID = 1
- 02-06 15:19:45.237 8938-8938/? I/Kathy: onStartCommand - startId = 1, Thread ID = 1
- 02-06 15:19:45.238 8938-8938/? I/Kathy: after StartService
分析:1.主线程打印出是1,所有回调方法中打印出的执行线程ID都是1,证明回调方法都是在主线程中执行的。2.三次调用startService,只触发一次onCreate回调,触发了三次onStartCommand回调,且startId分别为1,2,3。证明 多次startService不会重复执行onCreate回调,但每次都会执行onStartCommand回调。
第二种方式:通过bindService启动Service
bindService启动服务特点:1.bindService启动的服务和调用者之间是典型的client-server模式。调用者是client,service则是server端。service只有一个,但绑定到service上面的client可以有一个或很多个。这里所提到的client指的是组件,比如某个Activity。2.client可以通过IBinder接口获取Service实例,从而实现在client端直接调用Service中的方法以实现灵活交互,这在通过startService方法启动中是无法实现的。3.bindService启动服务的生命周期与其绑定的client息息相关。当client销毁时,client会自动与Service解除绑定。当然,client也可以明确调用Context的unbindService()方法与Service解除绑定。当没有任何client与Service绑定时,Service会自行销毁。
bindService代码实例
交互界面设计如下:
ActivityA界面布局.png
ActivityB界面布局.png
1.创建一个TestTwoService继承Service(Server)2.创建ActivityA,可以通过bindService绑定服务(client)3.创建ActivityB,可以通过bindService绑定服务(client)4.ActivityA可以跳转到ActivityB
TestTwoService创建如下:要想让Service支持bindService调用方式,需要做以下事情:1.在Service的onBind()方法中返回IBinder类型的实例。2.onBInd()方法返回的IBinder的实例需要能够返回Service实例本身。通常,最简单的方法就是在service中创建binder的内部类,加入类似getService()的方法返回Service,这样绑定的client就可以通过getService()方法获得Service实例了。
[java] view plain copy
- /**
- * Created by Kathy on 17-2-6.
- */
- public class TestTwoService extends Service{
- //client 可以通过Binder获取Service实例
- public class MyBinder extends Binder {
- public TestTwoService getService() {
- return TestTwoService.this;
- }
- }
- //通过binder实现调用者client与Service之间的通信
- private MyBinder binder = new MyBinder();
- private final Random generator = new Random();
- @Override
- public void onCreate() {
- Log.i("Kathy","TestTwoService - onCreate - Thread = " + Thread.currentThread().getName());
- super.onCreate();
- }
- @Override
- public int onStartCommand(Intent intent, int flags, int startId) {
- Log.i("Kathy", "TestTwoService - onStartCommand - startId = " + startId + ", Thread = " + Thread.currentThread().getName());
- return START_NOT_STICKY;
- }
- @Nullable
- @Override
- public IBinder onBind(Intent intent) {
- Log.i("Kathy", "TestTwoService - onBind - Thread = " + Thread.currentThread().getName());
- return binder;
- }
- @Override
- public boolean onUnbind(Intent intent) {
- Log.i("Kathy", "TestTwoService - onUnbind - from = " + intent.getStringExtra("from"));
- return false;
- }
- @Override
- public void onDestroy() {
- Log.i("Kathy", "TestTwoService - onDestroy - Thread = " + Thread.currentThread().getName());
- super.onDestroy();
- }
- //getRandomNumber是Service暴露出去供client调用的公共方法
- public int getRandomNumber() {
- return generator.nextint();
- }
- }
client端要做的事情:
1.创建ServiceConnection类型实例,并重写onServiceConnected()方法和onServiceDisconnected()方法。
2.当执行到onServiceConnected回调时,可通过IBinder实例得到Service实例对象,这样可实现client与Service的连接。
3.onServiceDisconnected回调被执行时,表示client与Service断开连接,在此可以写一些断开连接后需要做的处理。
创建ActivityA,代码如下:
[java] view plain copy
- /**
- * Created by Kathy on 17-2-6.
- */
- public class ActivityA extends Activity implements Button.OnClickListener {
- private TestTwoService service = null;
- private boolean isBind = false;
- private ServiceConnection conn = new ServiceConnection() {
- @Override
- public void onServiceConnected(componentname name, IBinder binder) {
- isBind = true;
- TestTwoService.MyBinder myBinder = (TestTwoService.MyBinder) binder;
- service = myBinder.getService();
- Log.i("Kathy", "ActivityA - onServiceConnected");
- int num = service.getRandomNumber();
- Log.i("Kathy", "ActivityA - getRandomNumber = " + num);
- }
- @Override
- public void onServiceDisconnected(ComponentName name) {
- isBind = false;
- Log.i("Kathy", "ActivityA - onServiceDisconnected");
- }
- };
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_a);
- Log.i("Kathy", "ActivityA - onCreate - Thread = " + Thread.currentThread().getName());
- findViewById(R.id.btnBindService).setOnClickListener(this);
- findViewById(R.id.btnUnbindService).setOnClickListener(this);
- findViewById(R.id.btnStartActivityB).setOnClickListener(this);
- findViewById(R.id.btnFinish).setOnClickListener(this);
- }
- @Override
- public void onClick(View v) {
- if (v.getId() == R.id.btnBindService) {
- //单击了“bindService”按钮
- Intent intent = new Intent(this, TestTwoService.class);
- intent.putExtra("from", "ActivityA");
- Log.i("Kathy", "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityA 执行 bindService");
- bindService(intent, conn, BIND_AUTO_CREATE);
- } else if (v.getId() == R.id.btnUnbindService) {
- //单击了“unbindService”按钮
- if (isBind) {
- Log.i("Kathy",
- "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityA 执行 unbindService");
- unbindService(conn);
- }
- } else if (v.getId() == R.id.btnStartActivityB) {
- //单击了“start ActivityB”按钮
- Intent intent = new Intent(this, ActivityB.class);
- Log.i("Kathy",
- "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityA 启动 ActivityB");
- startActivity(intent);
- } else if (v.getId() == R.id.btnFinish) {
- //单击了“Finish”按钮
- Log.i("Kathy",
- "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityA 执行 finish");
- this.finish();
- }
- }
- @Override
- protected void onDestroy() {
- super.onDestroy();
- Log.i("Kathy", "ActivityA - onDestroy");
- }
- }
创建ActivityB,代码如下:
[java] view plain copy
- /**
- * Created by Kathy on 17-2-6.
- */
- public class ActivityB extends Activity implements Button.OnClickListener {
- private TestTwoService service = null;
- private boolean isBind = false;
- private ServiceConnection conn = new ServiceConnection() {
- @Override
- public void onServiceConnected(ComponentName name, IBinder binder) {
- isBind = true;
- TestTwoService.MyBinder myBinder = (TestTwoService.MyBinder)binder;
- service = myBinder.getService();
- Log.i("Kathy", "ActivityB - onServiceConnected");
- int num = service.getRandomNumber();
- Log.i("Kathy", "ActivityB - getRandomNumber = " + num);
- }
- @Override
- public void onServiceDisconnected(ComponentName name) {
- isBind = false;
- Log.i("Kathy", "ActivityB - onServiceDisconnected");
- }
- };
- @Override
- protected void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.activity_b);
- findViewById(R.id.btnBindService).setOnClickListener(this);
- findViewById(R.id.btnUnbindService).setOnClickListener(this);
- findViewById(R.id.btnFinish).setOnClickListener(this);
- }
- @Override
- public void onClick(View v) {
- if(v.getId() == R.id.btnBindService){
- //单击了“bindService”按钮
- Intent intent = new Intent(this, TestTwoService.class);
- intent.putExtra("from", "ActivityB");
- Log.i("Kathy", "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityB 执行 bindService");
- bindService(intent, conn, BIND_AUTO_CREATE);
- }else if(v.getId() == R.id.btnUnbindService){
- //单击了“unbindService”按钮
- if(isBind){
- Log.i("Kathy", "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityB 执行 unbindService");
- unbindService(conn);
- }
- }else if(v.getId() == R.id.btnFinish){
- //单击了“Finish”按钮
- Log.i("Kathy", "----------------------------------------------------------------------");
- Log.i("Kathy", "ActivityB 执行 finish");
- this.finish();
- }
- }
- @Override
- public void onDestroy(){
- super.onDestroy();
- Log.i("Kathy", "ActivityB - onDestroy");
- }
- }
测试步骤1
step1: 点击ActivityA的bindService按钮
step2: 再点击ActivityA的unbindService按钮
Log输出:
[java] view plain copy
- 02-07 14:09:38.031 1738-1738/com.demo.kathy.demo I/Kathy: ActivityA - onCreate - Thread = main
- 02-07 14:09:39.488 1738-1738/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:09:39.488 1738-1738/com.demo.kathy.demo I/Kathy: ActivityA 执行 bindService
- 02-07 14:09:39.496 1738-1738/com.demo.kathy.demo I/Kathy: TestTwoService - onCreate - Thread = main
- 02-07 14:09:39.497 1738-1738/com.demo.kathy.demo I/Kathy: TestTwoService - onBind - Thread = main
- 02-07 14:09:39.500 1738-1738/com.demo.kathy.demo I/Kathy: ActivityA - onServiceConnected
- 02-07 14:09:39.500 1738-1738/com.demo.kathy.demo I/Kathy: ActivityA - getRandomNumber = -1046987376
- 02-07 14:09:50.866 1738-1738/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:09:50.866 1738-1738/com.demo.kathy.demo I/Kathy: ActivityA 执行 unbindService
- 02-07 14:09:50.870 1738-1738/com.demo.kathy.demo I/Kathy: TestTwoService - onUnbind - from = ActivityA
- 02-07 14:09:50.871 1738-1738/com.demo.kathy.demo I/Kathy: TestTwoService - onDestroy - Thread = main
总结调用bindService之后发生的事情:1.client执行bindService()2.如果Service不存在,则Service执行onCreate(),onBind()3.client实例ServiceConnection执行onServiceConnected()方法
总结调用unbindService之后发生的事情:1.client执行unbindService()2.client与Service解除绑定连接状态3.Service检测是否还有其他client与其连接,如果没有Service执行onUnbind()和onDestroy()
测试步骤2
step1: 点击ActivityA的bindService按钮step2: 再点击ActivityA的Finish按钮Log 输出:
[java] view plain copy
- 02-07 14:49:16.626 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - onCreate - Thread = main
- 02-07 14:49:18.102 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:49:18.102 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA 执行 bindService
- 02-07 14:49:18.105 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onCreate - Thread = main
- 02-07 14:49:18.110 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onBind - Thread = main
- 02-07 14:49:18.112 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - onServiceConnected
- 02-07 14:49:18.112 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - getRandomNumber = -318399886
- 02-07 14:49:19.540 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:49:19.540 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA 执行 finish
- 02-07 14:49:19.789 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - onDestroy
- 02-07 14:49:19.798 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onUnbind - from = ActivityA
- 02-07 14:49:19.798 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onDestroy - Thread = main
总结:如果client销毁,那么client会自动与Service解除绑定。
测试步骤3
step1: 点击ActivityA的bindService按钮
step2: 点击ActivityA的startActivity B按钮,切换到ActivityB
step3: 点击ActivityB中的bindService按钮
step4: 点击ActivityB中的unbindService按钮
step5: 点击ActivityB中的Finish按钮
step6: 点击ActivityA中的unbindService按钮
得到Log:
[java] view plain copy
- 02-07 14:55:04.390 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - onCreate - Thread = main
- 02-07 14:55:08.191 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:08.191 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA 执行 bindService
- 02-07 14:55:08.197 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onCreate - Thread = main
- 02-07 14:55:08.198 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onBind - Thread = main
- 02-07 14:55:08.205 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - onServiceConnected
- 02-07 14:55:08.205 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA - getRandomNumber = -706215542
- 02-07 14:55:23.261 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:23.261 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA 启动 ActivityB
- 02-07 14:55:29.239 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:29.239 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB 执行 bindService
- 02-07 14:55:29.241 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB - onServiceConnected
- 02-07 14:55:29.241 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB - getRandomNumber = 1827572726
- 02-07 14:55:33.951 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:33.951 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB 执行 unbindService
- 02-07 14:55:36.645 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:36.645 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB 执行 finish
- 02-07 14:55:36.852 12566-12566/com.demo.kathy.demo I/Kathy: ActivityB - onDestroy
- 02-07 14:55:43.137 12566-12566/com.demo.kathy.demo I/Kathy: ----------------------------------------------------------------------
- 02-07 14:55:43.137 12566-12566/com.demo.kathy.demo I/Kathy: ActivityA 执行 unbindService
- 02-07 14:55:43.143 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onUnbind - from = ActivityA
- 02-07 14:55:43.143 12566-12566/com.demo.kathy.demo I/Kathy: TestTwoService - onDestroy - Thread = main
总结bindService的生命周期:
1.点击ActivityA的bindService按钮
第一次调用bindService会实例化TestTwoService,然后执行其onBind()方法,得到IBinder类型的实例,将其作为参数传入ActivityA的ServiceConnection的onServiceConnected方法中,标志着ActivityA与TestTwoService建立了绑定。
2.点击ActivityB中的bindService按钮
由于TestTwoService已处于运行状态,所以再次调用bindService不会重新创建它的实例,所以也不会执行TestTwoService的onCreate()方法和onBind()方法。ActivityB与ActivityA共享IBinder实例。此时有两个client与TestTwoService绑定。
3.点击ActivityB中的unbindService按钮
ActivityB与TestTwoService解除了绑定,当没有任何client与Service绑定时,才会执行Service的onUnbind()方法。此时,ActivityA还在绑定连接中,所以不会执行Service的解绑方法。
4.点击ActivityA中的unbindService按钮
ActivityA执行unbindService之后,ActivityA与TestTwoService就解除绑定了,这样就没有client与TestTwoService绑定,这时候Android会销毁TestTwoService,在销毁前会先执行TestTwoService的onUnbind()方法,然后才会执行其onDestroy()方法,这样TestService就销毁了。
如何保证Service不被杀死?
1. onStartCommand方式中,返回START_STICKY
首先我们来看看onStartCommand都可以返回哪些值:
调用Context.startService方式启动Service时,如果Android面临内存匮乏,可能会销毁当前运行的Service,待内存充足时可以重建Service。而Service被Android系统强制销毁并再次重建的行为依赖于Service的onStartCommand()方法的返回值。
START_NOT_STICKY
如果返回START_NOT_STICKY,表示当Service运行的进程被Android系统强制杀掉之后,不会重新创建该Service。当然如果在其被杀掉之后一段时间又调用了startService,那么该Service又将被实例化。那什么情境下返回该值比较恰当呢?
如果我们某个Service执行的工作被中断几次无关紧要或者对Android内存紧张的情况下需要被杀掉且不会立即重新创建这种行为也可接受,那么我们便可将 onStartCommand的返回值设置为START_NOT_STICKY。
举个例子,某个Service需要定时从服务器获取最新数据:通过一个定时器每隔指定的N分钟让定时器启动Service去获取服务端的最新数据。当执行到Service的onStartCommand时,在该方法内再规划一个N分钟后的定时器用于再次启动该Service并开辟一个新的线程去执行网络操作。假设Service在从服务器获取最新数据的过程中被Android系统强制杀掉,Service不会再重新创建,这也没关系,因为再过N分钟定时器就会再次启动该Service并重新获取数据。
START_STICKY
如果返回START_STICKY,表示Service运行的进程被Android系统强制杀掉之后,Android系统会将该Service依然设置为started状态(即运行状态),但是不再保存onStartCommand方法传入的intent对象,然后Android系统会尝试再次重新创建该Service,并执行onStartCommand回调方法,但是onStartCommand回调方法的Intent参数为null,也就是onStartCommand方法虽然会执行但是获取不到intent信息。如果你的Service可以在任意时刻运行或结束都没什么问题,而且不需要intent信息,那么就可以在onStartCommand方法中返回START_STICKY,比如一个用来播放背景音乐功能的Service就适合返回该值。
- START_REDELIVER_INTENT
如果返回START_REDELIVER_INTENT,表示Service运行的进程被Android系统强制杀掉之后,与返回START_STICKY的情况类似,Android系统会将再次重新创建该Service,并执行onStartCommand回调方法,但是不同的是,Android系统会再次将Service在被杀掉之前最后一次传入onStartCommand方法中的Intent再次保留下来并再次传入到重新创建后的Service的onStartCommand方法中,这样我们就能读取到intent参数。只要返回START_REDELIVER_INTENT,那么onStartCommand重的intent一定不是null。如果我们的Service需要依赖具体的Intent才能运行(需要从Intent中读取相关数据信息等),并且在强制销毁后有必要重新创建运行,那么这样的Service就适合返回START_REDELIVER_INTENT。
2.提高Service的优先级
在AndroidManifest.xml文件中对于intent-filter可以通过android:priority = "1000"这个属性设置最高优先级,1000是最高值,如果数字越小则优先级越低,同时适用于广播。
3.提升Service进程的优先级
当系统进程空间紧张时,会依照优先级自动进行进程的回收。
Android将进程分为6个等级,按照优先级由高到低依次为:
- 前台进程foreground_app
- 可视进程visible_app
- 次要服务进程secondary_server
- 后台进程hiddena_app
- 内容供应节点content_provider
- 空进程empty_app
可以使用startForeground将service放到前台状态,这样低内存时,被杀死的概率会低一些。
4.在onDestroy方法里重启Service
当service走到onDestroy()时,发送一个自定义广播,当收到广播时,重新启动service。
5.系统广播监听Service状态
6.将APK安装到/system/app,变身为系统级应用
FROM:
Android Service两种启动方式详解(总结版)
http://www.jianshu.com/p/4c798c91a613
Android Service完全解析,关于服务你所需知道的一切(上)
http://blog.csdn.net/guolin_blog/article/details/11952435/
Android中Service的使用详解和注意点(LocalService)
http://www.cnblogs.com/linlf03/p/3296323.html
Android四大组件:Service服务史上最全面解析
http://www.jianshu.com/p/d963c55c3ab9
相关阅读
文章目录导读乐观锁CAS 原理ABA问题库表改造代码改造RedPacketDao新增接口方法及Mapper映射文件UserRedPacketServic接口及实现类
wifi调试 手机数据线插电脑上,用android studio做IDE,手机和电脑连接到同一个wifi。下面的xxx.xxx.x.xxx是手机ip,一般在wifi设置里
Android开发中libs和jinLibs文件夹的作用详解
前言 相信各位Android开发中们在Android 开发中经常和这两个文件夹打交道,以前一直迷迷糊糊的使用,没去想过。最近遇到了一些问题,仔
Detected problems with API 弹窗 屏蔽解决方案 andro
最近项目测试MM反馈每次进入app,会提示一个弹窗。 调查测试发现是android9.0系统上产生的,综合网上各位前被采坑的资料分享。先把
为了更好地维护交易秩序,保障消费者的合法权益,淘宝商城开店是需要去进行身份证认证的,很多的新手还不知道淘宝身份认证在哪,所以想知