直接new 一个Service对象,然后mService.setValue()不行吗?
不行,因为Service从创建到启动涉及到新进程创建和跨进程通信,应用层自己写任务量很大。
Service的启动流程:
- Process A进程采用Binder IPC向system_server进程发起startService请求;
- system_server进程接收到请求后,向zygote进程发送创建进程的请求;
- zygote进程fork出新的子进程Remote Service进程;
- Remote Service进程,通过Binder IPC向sytem_server进程发起attachApplication请求;
- system_server进程在收到请求后,进行一系列准备工作后,再通过binder IPC向remote Service进程发送scheduleCreateService请求;
- Remote Service进程的binder线程在收到请求后,通过handler向主线程发送CREATE_SERVICE消息;
主线程在收到Message后,通过发射机制创建目标Service,并回调Service.onCreate()方法。
到此,Service便正式启动完成。当创建的是本地服务或者服务所属进程已创建时,则无需经过上述步骤2、3,直接创建服务即可。 详细分析参考:startService过程源码分析
如何在Activity中获取一个已创建的Service对象?
Activity启动Service时,调用bindService (Intent service, ServiceConnection conn, int flags)方法,参数ServiceConnection是一个接口,共有两个抽象方法,其中 onServiceConnected(ComponentName name, IBinder service) 会给出IBinder类型的形参service,IBinder类也是一个接口,它的实现类是Binder.java,应用层可以继承Binder自定义自己的Binder类,所以IBinder具体是哪一个类,由Service的public IBinder onBind(Intent intent){} 方法的返回值决定。一般在Service内部自定义一个内部类MyBinder,这个MyBinder就可以获取Service的引用并传递出去。这样Activity就可以根据IBinder获取Service的引用了。
具体示例代码如下:
public interface OnProgressListener {
void onProgress(int progress);
}
package com.example.communication;
import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
public class MsgService extends Service {
public static final int MAX_PROGRESS = 100;
private int progress = 0;
private OnProgressListener onProgressListener;
public void setOnProgressListener(OnProgressListener onProgressListener) {
this.onProgressListener = onProgressListener;
}
public int getProgress() {
return progress;
}
public void startDownLoad(){
new Thread(new Runnable() {
@Override
public void run() {
while(progress < MAX_PROGRESS){
progress += 5;
if(onProgressListener != null){
onProgressListener.onProgress(progress);
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
}
@Override
public IBinder onBind(Intent intent) {
return new MsgBinder();
}
public class MsgBinder extends Binder{
public MsgService getService(){
return MsgService.this;
}
}
}
package com.example.communication;
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;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.ProgressBar;
public class MainActivity extends Activity {
private MsgService msgService;
private ProgressBar mProgressBar;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Intent intent = new Intent("com.example.communication.MSG_ACTION");
bindService(intent, conn, Context.BIND_AUTO_CREATE);
mProgressBar = (ProgressBar) findViewById(R.id.progressBar1);
Button mButton = (Button) findViewById(R.id.button1);
mButton.setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
msgService.startDownLoad();
}
});
}
ServiceConnection conn = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
msgService = ((MsgService.MsgBinder)service).getService();
msgService.setOnProgressListener(new OnProgressListener() {
@Override
public void onProgress(int progress) {
mProgressBar.setProgress(progress);
}
});
}
};
@Override
protected void onDestroy() {
unbindService(conn);
super.onDestroy();
}
}
Activity与Service通信的方式有哪些?
- Activity调用bindService (Intent service, ServiceConnection conn, int flags)方法,得到Service对象的一个引用,这样Activity可以直接调用到Service中的方法,如果要主动通知Activity,我们可以利用回调方法。
- Service向Activity发送消息,可以使用广播,当然Activity要注册相应的接收器。比如Service要向多个Activity发送同样的消息的话,用这种方法就更好。
使用广播方式,详细代码参考:Activity与Service通信的几种方式
Android之startService()和bindService()区别 start和bind同时使用时,service生命周期是什么
|