Android四大组件之Service(一)本地服务

Android中的服务和windows中的服务是类似的东西,服务一般没有用户操作界面,它运行于系统中不容易被用户发觉,可以使用它开发如监控之类的程序。

服务分为:本地服务与远程服务(AIDL进程间通信)。

服务的开发步骤,如下:

第一步:继承Service

public class MyService extends Service { }

第二步:在AndroidManifest.xml文件中的<application>节点里对服务进行配置,类似于Activity:

<service android:name=".MyService"/>

服务不能自己运行,需要通过调用Context.startService()Context.bindService()方法启动服务。这两个方法都可以启动Service,但是它们的使用场合有所不同。

它们的区别如下:


1.startService()

使用startService()方法启用服务,调用者与服务之间没有关连,即使调用者退出了,服务仍然运行。

采用Context.startService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onStart()方法。如果调用startService()方法前服务已经被创建,多次调用startService()方法并不会导致多次创建服务但会导致多次调用onStart()方法

采用startService()方法启动的服务,只能调用Context.stopService()方法结束服务,服务结束时会调用onDestroy()方法。

通过startService()stopService()启动关闭服务。适用于服务和Activity之间没有调用交互的情况。如果相互之间需要方法调用或者传递参数,需要使用bindService()unbindService()方法启动关闭服务。


2.bindService()

使用bindService()方法启用服务,调用者与服务绑定在了一起,调用者一旦退出,服务也就终止,大有“不求同时生,必须同时死”的特点。

采用Context.bindService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onBind()方法,这个时候调用者和服务绑定在一起。 如果客户端要与服务进行通信,那么,onBind()方法必须返回Ibinder对象。如果调用者退出了,系统就会先调用服务的onUnbind()方法,接着调用onDestroy()方法。如果调用bindService()方法前服务已经被绑定,多次调用bindService()方法并不会导致多次创建服务及绑定(也就是说onCreate()onBind()方法并不会被多次调用)。如果调用者希望与正在绑定的服务解除绑定,可以调用unbindService()方法,调用该方法也会导致系统调用服务的onUnbind()-->onDestroy()方法。

下面是一个电话监听service:


public class PhoneListenerService extends Service {
private static final String TAG = "PhoneListenerService";
private PhoneStateListener listener = new PhoneStateListener(){
	private String filename;
	private boolean recoding;
	private String mobile;
	private MediaRecorder recorder;
	@Override
	public void onCallStateChanged(int state, String incomingNumber) {
	   try {
		switch (state){
		case TelephonyManager.CALL_STATE_IDLE: /* 无任何状态时 */
			if(recorder!=null){
				  if(recoding){
				    	recorder.stop();
				    	recorder.release();
				    	Log.e(TAG, "record finish");
				    	recorder = null;
				   }
			}
			break;
				    	
		case TelephonyManager.CALL_STATE_OFFHOOK: /* 接起电话时 */
			SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMMddHHmmss");
			filename = mobile+ dateFormat.format(new Date()) + ".3gp";
			recorder = new MediaRecorder();
			recorder.setAudioSource(MediaRecorder.AudioSource.MIC);
			recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP);// .3pg
			recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
			recorder.setOutputFile("/mnt/sdcard/"+ filename);
			recorder.prepare();
			recorder.start();
			recoding = true;
			Log.e(TAG, "recording");
			break;	
				    	
		case TelephonyManager.CALL_STATE_RINGING: /* 电话进来时 */
			mobile = incomingNumber;
			break;
	        }
	     } catch (Exception e) {
		Log.e(TAG, e.toString());
	     }
	     super.onCallStateChanged(state, incomingNumber);
	}
};
	
@Override
public void onCreate() {
  super.onCreate();
  TelephonyManager telManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
  telManager.listen(listener, PhoneStateListener.LISTEN_CALL_STATE);
  Log.e(TAG, "onCreate()");
}
	
@Override
public IBinder onBind(Intent intent) {
	return null;
}
}

加入权限:


<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
 <!-- 在SDCard中创建与删除文件权限 -->
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
<!-- 往SDCard写入数据权限 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<!-- 访问internet权限 -->
<uses-permission android:name="android.permission.INTERNET"/>
<uses-permission android:name="android.permission.RECORD_AUDIO"/>


Activity与服务进行通信,开发人员通常把通信方法定义在接口里,然后让Ibinder对象实现该接口,而Activity通过该接口引用服务onBind()方法返回的Ibinder对象,然后调用Ibinder对象里自定义的通信方法。例子如下:

本例是一个本地服务,即服务与Activity在同一个应用内部。

接口:

public interface ICountService {
	public int getCount();
}

服务类:

public class CountService extends Service {
private boolean quit;
private int count;
private ServiceBinder serviceBinder = new ServiceBinder();

public class ServiceBinder extends Binder implements ICountService {
	@Override
	public int getCount() {
		return count;
	}
}
@Override
public IBinder onBind(Intent intent) {
	return serviceBinder;
}
@Override
public void onCreate() {
	super.onCreate();
	new Thread(new Runnable() {
		@Override
		public void run() {
			while (!quit) {
			    try {
				Thread.sleep(1000);
			    } catch (InterruptedException e) {}
			    count++;
			}
		}
	}).start();
}

@Override
public void onDestroy() {
	super.onDestroy();
	this.quit = true;
}
}
客户端Activity:
public class ClientActivity extends Activity {
	private ICountService countService;

	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		this.bindService(new Intent(this, CountService.class), this.serviceConnection, BIND_AUTO_CREATE);
	}

	@Override
	protected void onDestroy() {
		super.onDestroy();
		this.unbindService(serviceConnection);
	}	
	
	private ServiceConnection serviceConnection = new ServiceConnection() {
		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			countService = (ICountService) service;//对于本地服务,获取的实例和服务onBind()返回的实例是同一个
			int i = countService.getCount();			
			Log.v("CountService", "Count is " + i);
		}
		@Override
		public void onServiceDisconnected(ComponentName name) {
			countService = null;
		}
	};
}

服务的生命周期回调方法

与采用Context.startService()方法启动服务有关的生命周期方法

onCreate()-onStart()-onDestroy()

onCreate()该方法在服务被创建时调用,该方法只会被调用一次,无论调用多少次startService()bindService()方法,服务也只被创建一次。

onStart() 只有采用Context.startService()方法启动服务时才会回调该方法。该方法在服务开始运行时被调用。多次调用startService()方法尽管不会多次创建服务,但onStart() 方法会被多次调用。

onDestroy()该方法在服务被终止时调用。

l 与采用 Context.bindService () 方法启动服务有关的生命周期方法

onCreate()-onBind()-onUnbind()àonDestroy()

onBind()只有采用Context.bindService()方法启动服务时才会回调该方法。该方法在调用者与服务绑定时被调用,当调用者与服务已经绑定,多次调用Context.bindService()方法并不会导致该方法被多次调用。

onUnbind()只有采用Context.bindService()方法启动服务时才会回调该方法。该方法在调用者与服务解除绑定时被调用。

如果先采用startService()方法启动服务,然后调用bindService()方法绑定到服务,再调用unbindService()方法解除绑定,最后调用bindService()方法再次绑定到服务,触发的生命周期方法如下:

onCreate()-onStart()-onBind()-onUnbind()[重载后的方法需返回true]-onRebind()


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值