怎么在活动中调用服务中的方法呢?有什么方法能让活动和服务关系更紧密一些呢?这是我们这篇文章需要考虑的问题,下面我们来看看代码:
public class MyService extends Service {
private DownloadBinder mBinder = new DownloadBinder();
class DownloadBinder extends Binder {
public void startDownload() {
Log.d("MyService", "startDownload executed");
}
public int getProgress() {
Log.d("MyService", "getProgress executed");
return 0;
}
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
public MyService() {
}
@Override
public void onCreate() {
super.onCreate();
Log.d("MyService", "onCreate executed");
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.d("MyService", "onStartCommand executed");
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
Log.d("MyService", "onDestroy executed");
}
}
可以看到,这里我新建了一个DownloadBinder类,并让它继承Binder,然后在它的内部提供两个模拟的方法,因为这两个方法并没有实现真正的功能,在这两个方法中分别打印一行日志。
接着,在MyService中创建了DownloadBinder的实例,并使用onBind()方法返回了这个实例。
现在来看下MainActivity的代码
MainActivity的布局代码如下:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
>
<Button
android:id="@+id/btn_bind_service"
android:text="绑定服务"
android:textSize="20sp"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<Button
android:id="@+id/btn_unbind_service"
android:text="解绑服务"
android:textSize="20sp"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
</LinearLayout>
这里就是两个按钮,分别用于绑定服务和解绑服务。
MainActivity的java代码如下:
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private Button btn_bindService;
private Button btn_unbindService;
private MyService.DownloadBinder downloadBinder;
private ServiceConnection connection = new ServiceConnection() {
// 绑定服务成功的时候调用
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
downloadBinder = (MyService.DownloadBinder) service;
downloadBinder.startDownload();
downloadBinder.getProgress();
}
// 解绑服务的时候调用
@Override
public void onServiceDisconnected(ComponentName name) {
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btn_bindService = findViewById(R.id.btn_bind_service);
btn_unbindService = findViewById(R.id.btn_unbind_service);
btn_bindService.setOnClickListener(this);
btn_unbindService.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_bind_service:
Intent bindIntent = new Intent(this, MyService.class);
bindService(bindIntent, connection, BIND_AUTO_CREATE); // 绑定服务
break;
case R.id.btn_unbind_service:
unbindService(connection); // 解绑服务
break;
default:
break;
}
}
}
现在来解析下这些代码的意思:
可以看到,这里我首先创建了一个ServiceConnection的匿名类,在里面重写了两个方法,这两个方法分别会在活动与服务成功绑定以及解除绑定的时候调用。
在onServiceConnected()方法中,通过向下转型得到DownloadBinder的实例,有了这个实例,我们就可以在活动中调用DownloadBinder中的任何public()方法了,这里是调用了DownloadBinder的startDownload()方法和getProgress()方法。
接下来我们看下bindService()这个方法,这个方法接收3个参数,第一个参数是Intent对象,第二个参数是ServiceConnection的实例,第三个参数是一个标志位,这里传入的是BIND_AUTO_CREATE,表示在活动和服务进行绑定后自动创建服务,这会使MyService中的onCreate()方法得到执行,但onStartCommand()方法不会被执行。
好了,现在运行一下程序,点击一下绑定服务按钮,打印的日志如下所示:
可以看到,首先是MyService中的onCreate()方法得到执行,然后startDownload()方法和getProgress()方法都得到了执行,说明我们成功的在活动中调用了服务中的方法。