开启服务的时候,如果我们是通过bindService来绑定服务并且要向服务传递数据,可以直接在Intent中设置bundle来达到效果,但是如果是我们需要从服务中返回一些数据到Activity中的时候,实现起来就有各种各样的方法,比如说使用回调,使用广播等等,今天说的是使用回调的方法。
新建一个工程,并编写一个服务:
public class MyService extends Service {
private boolean connecting = false;
private Callback callback; @Nullable
@Override
public IBinder onBind(Intent intent) {
return new Binder();
} public class Binder extends android.os.Binder {
public MyService getService() {
return MyService.this;
}
} @Override
public void onCreate() {
super.onCreate();
connecting = true;
new Thread(new Runnable() { @Override
public void run() {
int i = 0;
while (connecting == true) {
i++;
if (callback != null) {
callback.onDataChange(i + "");
}
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}).start();
} public void setCallback(Callback callback) {
this.callback = callback;
} public static interface Callback {
void onDataChange(String data);
} @Override
public void onDestroy() {
super.onDestroy();
connecting = false;
}
}
在服务中的onCreate方法中,我们打开了一个线程来模拟服务的运行,并在线程每隔1s中给私有变量i赋值递增,然后我们编写了一个公有的接口Callback,并且定义了一个该接口的私有成员,并且在onCreate方法中调用了接口里面的函数onDataChange。接下来我们自定义了一个Binder的子类并在这个类中定义了函数返回当前的这个Service,这里的目的就是在Activity中可以访问到这个Service的回调接口Callback并实现该接口的方法。
Activity代码如下:
public class MainActivity extends AppCompatActivity implements View.OnClickListener,
ServiceConnection { private TextView tvOut; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tvOut = (TextView) findViewById(R.id.tvOut);
findViewById(R.id.btnBindService).setOnClickListener(this);
} @Override
public void onClick(View v) {
bindService(new Intent(this, MyService.class), this, BIND_AUTO_CREATE);
} @Override
public void onServiceConnected(ComponentName name, IBinder service) {
MyService.Binder binder = (MyService.Binder) service;
MyService myService = binder.getService();
myService.setCallback(new MyService.Callback() {
@Override
public void onDataChange(String data) {
Message msg = new Message();
msg.obj = data;
handler.sendMessage(msg);
}
});
} @Override
public void onServiceDisconnected(ComponentName name) { } private Handler handler = new Handler() {
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
tvOut.setText(msg.obj.toString());
}
};
}
因为服务绑定后,会从onBind方法中返回一个Binder对象,这个对象会在onServiceConnectde方法中获取到,所以我们先从Binder对象中获取到我们从服务传递过来的MyService对象,然后调用MyService对象的setCallback方法来设置我们需要的处理逻辑,这里是把i的值打印出来,因为服务中开启了线程,所以这里也不能直接更新UI。
总结:回调机制是Java中的一个重要特性,在Android中使用到的地方很广泛,例如我们给按钮设定点击事件等。这里的回调,其实是通过在发送端定义回调接口,并且调用接口的回调方法,然后在接收端实现该接口的方法。只要接口被调用了,就会回调接收端的被实现了的方法,这样数据就能传递过来。
注意:开启的服务是在主线程中运行的,如果在服务中开启了线程,那么在子线程中就不能直接更新UI