转 Service与Activity之间通信的2种方式
版权声明:因为个人水平有限,文章中可能会出现错误,如果你觉得有描述不当、代码错误等内容或者有更好的实现方式,欢迎在评论区告诉我,即刻回复!最后,欢迎关注博主!谢谢 https://blog.csdn.net/weixin_41101173/article/details/79718718
一、前期基础知识储备
(1)Service基础知识部分——参见笔者《Service精炼详解》,文章对于Service的简介、分类、实现、注意事项均有比较详细的说明;
(2)为何要进行Secvice和Activity的通信?
常用的服务一般是普通服务,即是不可交互的后台服务,该服务在活动中启动,但是启动之后,活动基本就和服务没有什么关系了。确实如此,我们在普通服务里是用startService()方法来启动Service这个服务的,之后服务会一直处于运行状态,但具体运行的是什么逻辑,活动控制不了,活动并不知道服务到底做了什么,完成的如何。
但是在很多场景下,活动是需要和服务进行交互的,比如音乐播放界面,用户可以根据播放进度条掌握播放的进度,用户也可以自己根据歌词的进度选择调整整首歌的进度。
要实现上面所示的功能,就要选择服务的另外一种类型——可交互的后台服务。以最常见的后台下载,前台显示的操作为例。实现这个功能的思路是创建一个专门的Binder类来对下载进行管理。
二、上代码,通过Binder对象具体实现两个组件之间的交互
①Service代码段
public class MyService extends Service { public MyService() { } private DownloadBinder mBinder = new DownloadBinder(); class DownloadBinder extends Binder { public void startDownload() { Log.d("MyService", "startDownload executed"); }//在服务中自定义startDownload()方法,待会活动中调用此方法 public int getProgress() { Log.d("MyService", "getProgress executed"); return 0; }//在服务中自定义getProgress()方法,待会活动中调用此方法 } @Override public IBinder onBind(Intent intent) { return mBinder; }//普通服务的不同之处,onBind()方法不在打酱油,而是会返回一个实例
②Activity代码段
public class MainActivity extends AppCompatActivity implements View.OnClickListener{ private MyService.DownloadBinder downloadBinder; private ServiceConnection connection = new ServiceConnection() { //可交互的后台服务与普通服务的不同之处,就在于这个connection建立起了两者的联系 @Override public void onServiceDisconnected(ComponentName name) { } @Override public void onServiceConnected(ComponentName name, IBinder service) { downloadBinder = (MyService.DownloadBinder) service; downloadBinder.startDownload(); downloadBinder.getProgress(); }//onServiceConnected()方法关键,在这里实现对服务的方法的调用 }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); Button bindService = (Button) findViewById(R.id.bind_service); Button unbindService = (Button) findViewById(R.id.unbind_service); bindService.setOnClickListener(this); unbindService.setOnClickListener(this); } @Override public void onClick(View v) { switch (v.getId()) { case R.id.bind_service: Intent bindIntent = new Intent(this, MyService.class); bindService(bindIntent, connection, BIND_AUTO_CREATE); // 绑定服务和活动,之后活动就可以去调服务的方法了 break; case R.id.unbind_service: unbindService(connection); // 解绑服务,服务要记得解绑,不要造成内存泄漏 break; default: break; } } }
点击一下按钮,观察logcat中的打印日志,就可以看到活动调用了服务的startDownload()和getProgress()方法。
三、另一种实现通信的方式是——通过broadcast(广播)的形式
有兴趣的读者可以参考这篇文章,《Android Service与Activity之间通信的几种方式》
笔者在这里就不具体实现了,具体就是利用了广播的特性实现交互。