基础总结篇之四:Service完全解析



Service在Android中和Activity是属于同一级别上的组件,我们可以将他们认为是两个好哥们,Activity仪表不凡,迷倒万千少女,经常做一些公众人物角色,而Service一副彪悍的长相,但却身强力壮,常常在后台做一些搬运工的力气活,虽然有些累,但大家都不能失去他。

下面我们就围绕Service对其进行全面讲解:

1.Service生命周期

Service生命周期可以从两种启动Service的模式开始讲起,分别是context.startService()和context.bindService()

(1).startService的启动模式下的生命周期:当我们首次使用startService启动一个服务时,系统会实例化一个Service实例,依次调用其onCreate和onStartCommand方法,然后进入运行状态,此后,如果再使用startService启动服务时,不再创建新的服务对象,系统会自动找到刚才创建的Service实例,调用其onStart方法;如果我们想要停掉一个服务,可使用stopService方法,此时onDestroy方法会被调用,需要注意的是,不管前面使用了多个次startService,只需一次stopService,即可停掉服务。

(2).bindService启动模式下的生命周期:在这种模式下,当调用者首次使用bindService绑定一个服务时,系统会实例化一个Service实例,并一次调用其onCreate方法和onBind方法,然后调用者就可以和服务进行交互了,此后,如果再次使用bindService绑定服务,系统不会创建新的Service实例,也不会再调用onBind方法;如果我们需要解除与这个服务的绑定,可使用unbindService方法,此时onUnbind方法和onDestroy方法会被调用。

两种模式有以下几点不同之处:startService模式下调用者与服务无必然联系,即使调用者结束了自己的生命周期,只要没有使用stopService方法停止这个服务,服务仍会运行;通常情况下,bindService模式下服务是与调用者生死与共的,在绑定结束之后,一旦调用者被销毁,服务也就立即终止,就像江湖上的一句话:不求同生,但愿同死。

值得一提的是,以前我们在使用startService启动服务时都是习惯重写onStart方法,在Android2.0时系统引进了onStartCommand方法取代onStart方法,为了兼容以前的程序,在onStartCommand方法中其实调用了onStart方法,不过我们最好是重写onStartCommand方法。

以上两种模式的流程如下图所示:



什么是Service?

         解惑:

                1、  Service不是分离开的进程,除非其他特殊情况,它不会运行在自己的进程,而是作为启动运行它的进程的一部分。

                2、  Service不是线程,这意味着它将在主线程里劳作。

        启动service有两种方法:

             1、  Context.startService()

                    调用者与服务之间没有关联,即使调用者退出,服务仍可运行

             2、  Context.bindService()

                    调用者与服务绑定在一起,调用者一旦退出,服务也就终止

Service的生命周期

         如果使用startService()启动service,系统将通过传入的Intent在底层搜索相关符合Intent里面信息的service。如果服务没有启动则先运行onCreate,然后运行onStartCommand (可在里面处理启动时传过来的Intent和其他参数),直到明显调用stopService或者stopSelf才将停止Service。无论运行startService多少次,只要调用一次stopService或者stopSelf,Service都会停止。使用stopSelf(int)方法可以保证在处理好intent后再停止。

         控制service运行的主要方式有两种,主要是根据onStartCommand方法返回的数值。方法:

         1、START_STICKY

         2、START_NOT_STICKY or START_REDELIVER_INTENT

         这里主要解释这三个变量的意义:

          1、  START_STICKY

                 在运行onStartCommand后service进程被kill后,那将保留在开始状态,但是不保留那些传入的intent。不久后service就会再次尝试重新创建,因为保留在开始状态,在创建     service后将保证调用onstartCommand。如果没有传递任何开始命令给service,那将获取到null的intent。 适用场景:后台播放音乐。这种运行模式的特点是需要显示启动并停止Service。

          2、  START_NOT_STICKY

                 在运行onStartCommand后service进程被kill后,并且没有新的intent传递给它。Service将移出开始状态,并且直到新的明显的方法(startService)调用才重新创建。因为如果没有传递任何未决定的intent那么service是不会启动,也就是期间onstartCommand不会接收到任何null的intent。适用场景:网上下载数据。

           3、  START_REDELIVER_INTENT

                在运行onStartCommand后service进程被kill后,系统将会再次启动service,并传入最后一个intent给onstartCommand。直到调用stopSelf(int)才停止传递intent。如果在被kill后还有未处理好的intent,那被kill后服务还是会自动启动。因此onstartCommand不会接收到任何null的intent。适用场景:关键业务处理。种模式保证了传给它Intent一定会被处理完毕

          客户端也可以使用bindService来保持跟service持久关联。谨记:如果使用这种方法,那么将不会调用onstartCommand(跟startService不一样,下面例子注释也有解析,大家可试试)。客户端将会在onBind回调中接收到IBinder接口返回的对象。通常IBinder作为一个复杂的接口通常是返回aidl数据。

         Service也可以混合start和bind一起使用。

权限

         要运行service,首先必须在AndroidManifest.xml里申明<service>标签。

         Service能够保护个人的IPC调用,所以在执行实现该调用时前先使用checkCallingPermission(String) 方法检查是否有这个权限。

 

进程生命周期

         当service运行在低内存的环境时,将会kill掉一下存在的进程。因此进程的优先级将会很重要:

         1、  如果service当前正在执行onCreate、onStartCommand、onDestroy方法,主进程将会成为前台进程来保证代码可以执行完成避免被kill

         2、  如果service已经启动了,那么主进程将会比其他可见的进程的重要性低,但比其他看不见的进程高。因为只有少部分进程始终是用户可见的,因此除非在极度低内存的时候,不然 service是不会被kill的。

         3、  如果有客户端关联到service,那么service永远比客户端重要。也就是说客户端可见,那么service也可见(我理解这里的可见并不是可以看到,而是重要性,因为可见往往就表示重要性高)。

         4、  Service可以使用startForeground API将service放到前台状态。这样在低内存时被kill的几率更低,但是文档后面又写了,如果在极度极度低内存的压力下,该service理论上还是会被kill掉。但这个情况基本不用考虑。

         当然如果service怎么保持还是被kill了,那你可以通过重写onStartCommand返回变量来设置它的启动方式。比如:START_STICKY、START_REDELIVER_INTENT等等,前面已经讨论了它们的作用,这里就不再累赘了

        另外:

        service 的onCreate和onStartCommand 是运行在主线程的,所以如果里面有处理耗时间的任务。两种处理:

        1、  请将它们都挪到新的线程里。

        2、  用系统提供的IntentService,它继承了Service,它处理数据是用自身新开的线程。




下面我们就结合实例来演示一下这两种模式的生命周期过程。我们新建一个名为service的项目,然后创建一个MyService的服务类,代码如下:

  1. package com.scott.service;  
  2.   
  3. import android.app.Service;  
  4. import android.content.Intent;  
  5. import android.os.IBinder;  
  6. import android.util.Log;  
  7.   
  8. public class MyService extends Service {  
  9.   
  10.     private static final String TAG = "MyService";  
  11.       
  12.     @Override  
  13.     public void onCreate() {  
  14.         super.onCreate();  
  15.         Log.i(TAG, "onCreate called.");  
  16.     }  
  17.       
  18.     @Override  
  19.     public int onStartCommand(Intent intent, int flags, int startId) {  
  20.         Log.i(TAG, "onStartCommand called.");  
  21.         return super.onStartCommand(intent, flags, startId);  
  22.     }  
  23.       
  24.     @Override  
  25.     public void onStart(Intent intent, int startId) {  
  26.         super.onStart(intent, startId);  
  27.         Log.i(TAG, "onStart called.");  
  28.     }  
  29.       
  30.     @Override  
  31.     public IBinder onBind(Intent intent) {  
  32.         Log.i(TAG, "onBind called.");  
  33.         return null;  
  34.     }  
  35.       
  36.     @Override  
  37.     public boolean onUnbind(Intent intent) {  
  38.         Log.i(TAG, "onUnbind called.");  
  39.         return super.onUnbind(intent);  
  40.     }  
  41.       
  42.     @Override  
  43.     public void onDestroy() {  
  44.         super.onDestroy();  
  45.         Log.i(TAG, "onDestroy called.");  
  46.     }  
  47. }  
package com.scott.service;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.util.Log;

public class MyService extends Service {

	private static final String TAG = "MyService";
	
	@Override
	public void onCreate() {
		super.onCreate();
		Log.i(TAG, "onCreate called.");
	}
	
	@Override
	public int onStartCommand(Intent intent, int flags, int startId) {
		Log.i(TAG, "onStartCommand called.");
		return super.onStartCommand(intent, flags, startId);
	}
	
	@Override
	public void onStart(Intent intent, int startId) {
		super.onStart(intent, startId);
		Log.i(TAG, "onStart called.");
	}
	
	@Override
	public IBinder onBind(Intent intent) {
		Log.i(TAG, "onBind called.");
		return null;
	}
	
	@Override
	public boolean onUnbind(Intent intent) {
		Log.i(TAG, "onUnbind called.");
		return super.onUnbind(intent);
	}
	
	@Override
	public void onDestroy() {
		super.onDestroy();
		Log.i(TAG, "onDestroy called.");
	}
}
然后再AndroidManifest.xml中配置服务信息,不然这个服务就不会生效,配置如下:

  1. <service android:name=".MyService">  
  2.             <intent-filter>  
  3.                  <action android:name="android.intent.action.MyService" />  
  4.                  <category android:name="android.intent.category.DEFAULT" />  
  5.             </intent-filter>  
  6.         </service>  
<service android:name=".MyService">
        	<intent-filter>
				 <action android:name="android.intent.action.MyService" />
                 <category android:name="android.intent.category.DEFAULT" />
			</intent-filter>
        </service>
如果服务只是在本应用中使用,大可以去掉<intent-filter>属性。

服务搭建完成之后,我们就来关注一下调用者MainActivity,它很简单,只有两个按钮,一个是启动服务,另一个是停止服务,我们来看一下他们的点击事件:

  1. /** 
  2.  * 启动服务 
  3.  * @param view 
  4.  */  
  5. public void start(View view) {  
  6.     Intent intent = new Intent(this, MyService.class);  
  7.     startService(intent);  
  8. }  
  9.   
  10. /** 
  11.  * 停止服务 
  12.  * @param view 
  13.  */  
  14. public void stop(View view) {  
  15.     Intent intent = new Intent(this, MyService.class);  
  16.     stopService(intent);  
  17. }  
	/**
	 * 启动服务
	 * @param view
	 */
	public void start(View view) {
		Intent intent = new Intent(this, MyService.class);
		startService(intent);
	}
	
	/**
	 * 停止服务
	 * @param view
	 */
	public void stop(View view) {
		Intent intent = new Intent(this, MyService.class);
		stopService(intent);
	}
接下来我们就先点击一次启动按钮,看看都发生了些什么。日志打印结果如下:


当然我们觉得还不过瘾,再点击一次,我们会发现结果略有不同:


我们看到第二次点击时onCreate方法就不再被调用了,而是直接调用了onStartCommand方法(onStartCommand中又调用了onStart方法)。我们选择“Settings->Application s->Running services”就会发现我们刚刚启动的服务:


然后我们点击停止按钮,试图停止服务,我们发现如下现象:


我们会发现onDestroy方法被调用了,此时服务就停止运行了。我们再次查看“Running services”,就会发现MyService这个服务已全无踪迹。

在这个过程中,onBind方法和onUnbind方法始终没被调用,我们下面就让这两位show一下自己。

我们修改一下MainActivity的代码,使其可以可以以bindService的方式启动一个服务,代码如下:

  1. private ServiceConnection conn = new ServiceConnection() {  
  2.       
  3.     @Override  
  4.     public void onServiceConnected(ComponentName name, IBinder service) {  
  5.         //connected  
  6.         Log.i(TAG, "onServiceConnected called.");  
  7.     }  
  8.   
  9.     /** 
  10.      *  Called when a connection to the Service has been lost. 
  11.      *  This typically happens when the process hosting the service has crashed or been killed. 
  12.      *  This does not remove the ServiceConnection itself. 
  13.      *  this binding to the service will remain active, 
  14.      *  and you will receive a call to onServiceConnected when the Service is next running. 
  15.      */  
  16.     @Override  
  17.     public void onServiceDisconnected(ComponentName name) {  
  18.     }  
  19. };  
  20.   
  21. /** 
  22.  * 绑定服务 
  23.  * @param view 
  24.  */  
  25. public void bind(View view) {  
  26.     Intent intent = new Intent(this, MyService.class);  
  27.     bindService(intent, conn, Context.BIND_AUTO_CREATE);  
  28. }  
  29.   
  30. /** 
  31.  * 解除绑定 
  32.  * @param view 
  33.  */  
  34. public void unbind(View view) {  
  35.     unbindService(conn);  
  36. }  
	private ServiceConnection conn = new ServiceConnection() {
		
		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			//connected
			Log.i(TAG, "onServiceConnected called.");
		}

		/**
		 * 	Called when a connection to the Service has been lost.
		 *	This typically happens when the process hosting the service has crashed or been killed.
		 *	This does not remove the ServiceConnection itself.
		 *	this binding to the service will remain active,
		 *	and you will receive a call to onServiceConnected when the Service is next running.
		 */
		@Override
		public void onServiceDisconnected(ComponentName name) {
		}
	};
	
	/**
	 * 绑定服务
	 * @param view
	 */
	public void bind(View view) {
		Intent intent = new Intent(this, MyService.class);
		bindService(intent, conn, Context.BIND_AUTO_CREATE);
	}
	
	/**
	 * 解除绑定
	 * @param view
	 */
	public void unbind(View view) {
		unbindService(conn);
	}
在使用bindService绑定服务时,我们需要一个ServiceConnection代表与服务的连接,它只有两个方法,onServiceConnected和onServiceDisconnected,前者是在操作者在连接一个服务成功时被调用,而后者是在服务崩溃或被杀死导致的连接中断时被调用,而如果我们自己解除绑定时则不会被调用,所以我们这里只研究onServiceConnected这个方法。

看样子是可以去绑定一个服务了,其实还不行,因为我们前面服务中的onBind方法返回值为null,这样是不行的,要想实现绑定操作,必须返回一个实现了IBinder接口类型的实例,该接口描述了与远程对象进行交互的抽象协议,有了它我们才能与服务进行交互。我们于是有了这样的代码:

  1. @Override  
  2. public IBinder onBind(Intent intent) {  
  3.     Log.i(TAG, "onBind called.");  
  4.     return new Binder() {};  
  5. }  
	@Override
	public IBinder onBind(Intent intent) {
		Log.i(TAG, "onBind called.");
		return new Binder() {};
	}
我们返回了一个Binder的实例,而这个Binder恰恰是实现了IBinder接口,所以这样就可以实现绑定服务的操作了,一起来演示一下。

先点击一下绑定按钮,我们会发现在MainActivity中打印日志如下:


似的,onServiceConnected方法被调用了,看来绑定连接已经成功了,看看MyService如何:


onCreate方法和onBind方法被调用了,此时服务已进入运行阶段,如果再次点击绑定按钮,onCreate和onBinder并不会再次被调用,这个过程中它们仅被调用一次。

然后点击解除绑定按钮,我们会发现MyService打印如下:


可以看到onUnbind方法和onDestroy方法被调用了,此时MyService已被销毁,整个生命周期结束。

另一方面,当我们退出MainActivity时,服务也会随之而结束,从这一点上看,MyService可以说是誓死追随着MainActivity。

需要注意的是,在连接中断状态再去做解除绑定操作会引起一个异常,在MainActivity销毁之前没有进行解除绑定也会导致后台出现异常信息,此时我们就要想办法确保不会出现此类情况,可以这样做:

  1. private boolean binded;  
  2.   
  3. @Override  
  4. public void onServiceConnected(ComponentName name, IBinder service) {  
  5.     binded = true;  
  6. }  
  7.   
  8. /** 
  9.  * 解除绑定 
  10.  * @param view 
  11.  */  
  12. public void unbind(View view) {  
  13.     unbindService();  
  14. }  
  15.   
  16. @Override  
  17. protected void onDestroy() {  
  18.     super.onDestroy();  
  19.     unbindService();  
  20. }  
  21.   
  22. /** 
  23.  * 解除服务绑定 
  24.  */  
  25. private void unbindService() {  
  26.     if (binded) {  
  27.         unbindService(conn);  
  28.         binded = false;  
  29.     }  
  30. }  
	private boolean binded;
	
	@Override
	public void onServiceConnected(ComponentName name, IBinder service) {
		binded = true;
	}
	
	/**
	 * 解除绑定
	 * @param view
	 */
	public void unbind(View view) {
		unbindService();
	}
	
	@Override
	protected void onDestroy() {
		super.onDestroy();
		unbindService();
	}
	
	/**
	 * 解除服务绑定
	 */
	private void unbindService() {
		if (binded) {
			unbindService(conn);
			binded = false;
		}
	}

以上就是bindService的生命周期,正如我们上面讲的一样,使用bindService启动服务后调用者和服务绑定到了一起,当调用者被销毁,服务也立即结终止。

通常情况下是这样的,不过也有特殊情况。当startService和bindService在同一场合下使用时,就会出现稍微不同的现象。

如果我们先以startService方式启动服务,然后再用bindService绑定到这个服务,之后使用unbindService解除绑定,此时服务并不会因此而终止,而是继续运行,直到我们使用stopService来停止这个服务。下面我们再修改一下代码以验证这个过程。MyService保持不变,我们只需修改一下MainActivity。MainActivity最新代码如下:

  1. package com.scott.service;  
  2.   
  3. import android.app.Activity;  
  4. import android.content.ComponentName;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7. import android.content.ServiceConnection;  
  8. import android.os.Bundle;  
  9. import android.os.IBinder;  
  10. import android.util.Log;  
  11. import android.view.View;  
  12.   
  13. public class MainActivity extends Activity {  
  14.       
  15.     private static final String TAG = "MainActivity";  
  16.       
  17.     @Override  
  18.     public void onCreate(Bundle savedInstanceState) {  
  19.         super.onCreate(savedInstanceState);  
  20.         setContentView(R.layout.main);  
  21.     }  
  22.       
  23.     private ServiceConnection conn = new ServiceConnection() {  
  24.           
  25.         @Override  
  26.         public void onServiceConnected(ComponentName name, IBinder service) {  
  27.             Log.i(TAG, "onServiceConnected called.");  
  28.         }  
  29.           
  30.         @Override  
  31.         public void onServiceDisconnected(ComponentName name) {  
  32.         }  
  33.     };  
  34.       
  35.     /** 
  36.      * 启动服务 
  37.      * @param view 
  38.      */  
  39.     public void start(View view) {  
  40.         Intent intent = new Intent(this, MyService.class);  
  41.         startService(intent);  
  42.     }  
  43.       
  44.     /** 
  45.      * 绑定服务 
  46.      * @param view 
  47.      */  
  48.     public void bind(View view) {  
  49.         Intent intent = new Intent(this, MyService.class);  
  50.         bindService(intent, conn, Context.BIND_AUTO_CREATE);  
  51.     }  
  52.       
  53.     /** 
  54.      * 解除绑定 
  55.      * @param view 
  56.      */  
  57.     public void unbind(View view) {  
  58.         unbindService(conn);  
  59.     }  
  60.       
  61.     /** 
  62.      * 停止服务 
  63.      * @param view 
  64.      */  
  65.     public void stop(View view) {  
  66.         Intent intent = new Intent(this, MyService.class);  
  67.         stopService(intent);  
  68.     }  
  69. }  
package com.scott.service;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.util.Log;
import android.view.View;

public class MainActivity extends Activity {
	
	private static final String TAG = "MainActivity";
	
	@Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
    }
	
	private ServiceConnection conn = new ServiceConnection() {
		
		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			Log.i(TAG, "onServiceConnected called.");
		}
		
		@Override
		public void onServiceDisconnected(ComponentName name) {
		}
	};
	
	/**
	 * 启动服务
	 * @param view
	 */
	public void start(View view) {
		Intent intent = new Intent(this, MyService.class);
		startService(intent);
	}
	
	/**
	 * 绑定服务
	 * @param view
	 */
	public void bind(View view) {
		Intent intent = new Intent(this, MyService.class);
		bindService(intent, conn, Context.BIND_AUTO_CREATE);
	}
	
	/**
	 * 解除绑定
	 * @param view
	 */
	public void unbind(View view) {
		unbindService(conn);
	}
	
	/**
	 * 停止服务
	 * @param view
	 */
	public void stop(View view) {
		Intent intent = new Intent(this, MyService.class);
		stopService(intent);
	}
}
在MainActivity中包含了四个按钮事件,分别是startService、bindService、unbindService和stopService,我们逐一地按下,看看都发生了什么。

首先按下启动服务的按钮,MyService打印如下:

恩,意料之中。然后我们再按下绑定服务的按钮,MyService打印如下:


此时,只有onBind被调用,之后两者就绑定成功。我们再按下解除绑定的按钮,MyService打印如下:


此时,onUnbind方法方法被调用,注意,此时MyService并没有因解除绑定而终止,而是继续运行。也许我们心里会问,如果多次按下绑定服务的按钮或重复以上两个步骤,结果如何呢?答案是onBind和onUnbind都不会再被调用了。看不到onBind被调用,是不是没有绑定成功啊,我们来看一下MainActivity打印信息:


重复按下绑定按钮,几次都绑定成功了。最后我们按下停止服务的按钮,MyService打印如下:


此时,onDestroy被调用了,此时MyService停止了运行,整个生命周期结束。

以上就是关于MyService生命周期的讲解,下面我们来介绍一下如何与服务进行通信。与服务之间的通信可以分为两种,进程内的通信和进程间的通信,前者调用者和服务在同一应用进程内,而后者是分布在不同应用进程中的。

2.进程内与服务通信

进程内与服务通信实际上就是通过bindService的方式与服务绑定,获取到通信中介Binder实例,然后通过调用这个实例的方法,完成对服务的各种操作。我们上面也介绍了不少关于bindService的内容,下面我们就针对实际需求对代码做改动。首先是MyService,代码如下:

  1. package com.scott.service;  
  2.   
  3. import android.app.Service;  
  4. import android.content.Intent;  
  5. import android.os.Binder;  
  6. import android.os.IBinder;  
  7. import android.util.Log;  
  8.   
  9. public class MyService extends Service {  
  10.   
  11.     private static final String TAG = "MyService";  
  12.       
  13.     @Override  
  14.     public IBinder onBind(Intent intent) {  
  15.         Log.i(TAG, "onBind called.");  
  16.         return new MyBinder();  
  17.     }  
  18.       
  19.     /** 
  20.      * 绑定对象 
  21.      * @author user 
  22.      * 
  23.      */  
  24.     public class MyBinder extends Binder {  
  25.           
  26.         /** 
  27.          * 问候 
  28.          * @param name 
  29.          */  
  30.         public void greet(String name) {  
  31.             Log.i(TAG, "hello, " + name);  
  32.         }  
  33.     }  
  34. }  
package com.scott.service;

import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;

public class MyService extends Service {

	private static final String TAG = "MyService";
	
	@Override
	public IBinder onBind(Intent intent) {
		Log.i(TAG, "onBind called.");
		return new MyBinder();
	}
	
	/**
	 * 绑定对象
	 * @author user
	 *
	 */
	public class MyBinder extends Binder {
		
		/**
		 * 问候
		 * @param name
		 */
		public void greet(String name) {
			Log.i(TAG, "hello, " + name);
		}
	}
}
我们创建了一个MyBinder的内部类,定义了一个greet方法,在onBind方法中就将这个MyBinder的实例返回,只要调用者获取到这个实例,就可以像拿着游戏手柄一样对服务进行操作。我们来看一下调用者的代码吧,MainActivity代码如下:

  1. package com.scott.service;  
  2.   
  3. import android.app.Activity;  
  4. import android.content.ComponentName;  
  5. import android.content.Context;  
  6. import android.content.Intent;  
  7. import android.content.ServiceConnection;  
  8. import android.os.Bundle;  
  9. import android.os.IBinder;  
  10. import android.view.View;  
  11.   
  12. public class MainActivity extends Activity {  
  13.       
  14.     /** 
  15.      * 绑定对象实例 
  16.      */  
  17.     private MyService.MyBinder binder;  
  18.       
  19.     @Override  
  20.     public void onCreate(Bundle savedInstanceState) {  
  21.         super.onCreate(savedInstanceState);  
  22.         setContentView(R.layout.main);  
  23.     }  
  24.       
  25.     private ServiceConnection conn = new ServiceConnection() {  
  26.           
  27.         @Override  
  28.         public void onServiceConnected(ComponentName name, IBinder service) {  
  29.             binder = (MyService.MyBinder) service;  //获取其实例  
  30.             binder.greet("scott");                  //调用其方法  
  31.         }  
  32.           
  33.         @Override  
  34.         public void onServiceDisconnected(ComponentName name) {  
  35.         }  
  36.     };  
  37.       
  38.     /** 
  39.      * 绑定服务 
  40.      * @param view 
  41.      */  
  42.     public void bind(View view) {  
  43.         Intent intent = new Intent(this, MyService.class);  
  44.         bindService(intent, conn, Context.BIND_AUTO_CREATE);  
  45.     }  
  46.       
  47.     /** 
  48.      * 解除绑定 
  49.      * @param view 
  50.      */  
  51.     public void unbind(View view) {  
  52.         unbindService(conn);  
  53.     }  
  54. }  
package com.scott.service;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.view.View;

public class MainActivity extends Activity {
	
	/**
	 * 绑定对象实例
	 */
	private MyService.MyBinder binder;
	
	@Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
    }
	
	private ServiceConnection conn = new ServiceConnection() {
		
		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			binder = (MyService.MyBinder) service;	//获取其实例
			binder.greet("scott");					//调用其方法
		}
		
		@Override
		public void onServiceDisconnected(ComponentName name) {
		}
	};
	
	/**
	 * 绑定服务
	 * @param view
	 */
	public void bind(View view) {
		Intent intent = new Intent(this, MyService.class);
		bindService(intent, conn, Context.BIND_AUTO_CREATE);
	}
	
	/**
	 * 解除绑定
	 * @param view
	 */
	public void unbind(View view) {
		unbindService(conn);
	}
}
在上面的代码中,我们是在绑定服务成功时将IBinder类型的service参数强转为MyService.MyBinder类型,获取绑定中介实例,然后调用其greet方法。

操作一下,看看效果如何。先点击绑定服务的按钮,MyService打印如下:


需要注意的是,与服务绑定是一个异步的过程,也就是说,在这一刻我们绑定服务,下一刻我们去操作binder对象,也许它还为null,这就容易引起空指针异常,正确的做法是把这些操作放到绑定成功之后,确保万无一失。

以上就是进程内通信的内容。

关于进程间通信的内容,限于篇幅原因,这里就不再陈述了,大家可以看我之前的两篇文章,做一下了解:

使用AIDL实现进程间的通信

使用AIDL实现进程间的通信之复杂类型传递


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值