介绍

在前一天我们介绍了Android中有两种启动Service的方法。并擅述了startService和bindService的区别。同时我们着重讲了startService。

因此今天我们就来讲bindService。bindService大家可以认为它是和Android的一个共生体。即这个service所属的activity如果消亡那么bindService也会消亡。

因此今天我们以一个比较复杂的例子,activity<->service间互相传值来讲透这个bindService的使用,同时我们在这个例子中故意留下一个坑即:在Service里使用Thread处理大事务是不是就一定安全呢?也不安全,它也会引起ANR即:Application Not Responding-安卓崩溃。从而以这个坑来引出IntentService的使用。

来看例子

我们设有三个按钮:

  • 【BIND SERVICE】-点击后运行Service
  • 【STOP BINDING】-点击后结束Service
  • 【GET VALUE FROM BINDER】-通过Activity获取正在BINDING的Service内的值,此处我们留下了一个ANR的坑,即获取Service内的值时我们留了一个Thread.Sleep(30000)的长事务,来观察ANR;

此处记得按钮的点击顺序为:先点【BIND SERVICE】->再点【GET VALUE FROM BINDER】->再点【STOP BINDING】不过此处你没有机会点这个【STOP BINDING】按钮,因为在GET时你已经ANR(崩溃)了。

来看全代码展示。

全代码

Service注册

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"xmlns:tools="http://schemas.android.com/tools"><applicationandroid:allowBackup="true"android:dataExtractionRules="@xml/data_extraction_rules"android:fullBackupContent="@xml/backup_rules"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:roundIcon="@mipmap/ic_launcher_round"android:supportsRtl="true"android:theme="@style/Theme.DemoBindService"tools:targetApi="31"><serviceandroid:name=".SampleBindService"android:enabled="true"android:exported="true"><intent-filter><actionandroid:name="org.mk.android.demo.SampleBindService"/></intent-filter></service><activityandroid:name=".MainActivity"android:exported="true"><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter><meta-dataandroid:name="android.app.lib_name"android:value="" /></activity></application></manifest>

Service类(坑来了)

package org.mk.android.demo;import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.util.Log;public class SampleBindService extends Service {private final String TAG = "SimpleBindService";private int count;private boolean quit;private CountNumBinder countNumBinder = new CountNumBinder();public SampleBindService() {}@Overridepublic IBinder onBind(Intent intent) {// TODO: Return the communication channel to the service.Log.i(TAG, ">>>>>>onBind方法被调用");return countNumBinder;}//Service被关闭前回调@Overridepublic void onDestroy() {super.onDestroy();this.quit = true;Log.i(TAG, ">>>>>>onDestroyed方法被调用!");}@Overridepublic void onRebind(Intent intent) {Log.i(TAG, ">>>>>>onRebind方法被调用!");super.onRebind(intent);}//Service被创建时调用@Overridepublic void onCreate() {Log.i(TAG, ">>>>>>onCreate方法被调用");super.onCreate();//创建一个线程动态地修改count的值new Thread() {public void run() {while (!quit) {try {Thread.sleep(1000);} catch (Exception e) {e.printStackTrace();}count++;}};}.start();}//Service断开连接时回调@Overridepublic boolean onUnbind(Intent intent) {Log.i(TAG, ">>>>>>onUnbind方法被调用!");return true;}//Service被启动时调用@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {Log.i(TAG, ">>>>>>onStartCommand方法被调用");return super.onStartCommand(intent, flags, startId);}public class CountNumBinder extends Binder {public int getCount() {Log.i(TAG, ">>>>>>into long waiting");try {Thread.sleep(300000);} catch (Exception e) {}return -1;}}
}

我们可以看到,这个Service以每秒对着count+1.

然后通过bindService的onBind生命体里以一个CountNumBinder暴露出去,给到外部可以通过一个getCount方法来调用获取Service里当前count的值,但是这个值在获取前我们会使用Thread.sleep(30000)-30秒来模拟ANR。

主运行类-MainActivity.java

  • 在调用Service的activity里我们使用bindService(intent, conn, Service.BIND_AUTO_CREATE);来启动。
  • 这边这个conn是一个ServiceConnection类,new出一个ServiceConnection类并覆盖里面的
    • onServiceConnected方法,用于接受bindService返回的对象;
    • onServiceDisconnected方法,用于在这个bindService被销毁时作处理;

具体代码如下:

package org.mk.android.demo;import androidx.appcompat.app.AppCompatActivity;import android.app.Service;
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.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;public class MainActivity extends AppCompatActivity {private final String TAG = "SimpleBindService";private Button buttonBindService;private Button buttonStopBinding;private Button buttonGetValueFromBinder;private Context ctx;private Intent intent;private SampleBindService.CountNumBinder countNumBinder;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);buttonBindService = (Button) findViewById(R.id.buttonBindService);buttonStopBinding = (Button) findViewById(R.id.buttonStopBinding);buttonGetValueFromBinder = (Button) findViewById(R.id.buttonGetValueFromBinder);ctx = MainActivity.this;intent = new Intent(ctx, SampleBindService.class);buttonBindService.setOnClickListener(new OnClickListener());buttonStopBinding.setOnClickListener(new OnClickListener());buttonGetValueFromBinder.setOnClickListener(new OnClickListener());}private ServiceConnection conn = new ServiceConnection() {//Activity与Service断开连接时回调该方法@Overridepublic void onServiceDisconnected(ComponentName name) {Log.i(TAG, ">>>>>>Service DisConnected");}//Activity与Service连接成功时回调该方法@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {Log.i(TAG, ">>>>>>Service Connected");countNumBinder = (SampleBindService.CountNumBinder) service;}};class OnClickListener implements View.OnClickListener {@Overridepublic void onClick(View view) {Intent eIntent;switch (view.getId()) {case R.id.buttonBindService:bindService(intent, conn, Service.BIND_AUTO_CREATE);break;case R.id.buttonStopBinding:unbindService(conn);break;case R.id.buttonGetValueFromBinder:Toast.makeText(getApplicationContext(), "Service的count" + "的值为:" + countNumBinder.getCount(), Toast.LENGTH_LONG).show();break;}}}
}

运行效果

  1. 先点【BIND SERVICE】;
  2. 再点【GET VALUE FROM BINDER】;

看,ANR出现了。

这就是我说的坑,怎么解决这个坑,请听下回分解。

Android入门第41天-Android中的Service(bindService)相关推荐

  1. Android入门第9天-Android读本地JSON文件并显示

    Android在大都情况会取得后台Service返回的json数据来做前端展示.那么Android是如何处理json文件的呢?我们就一起跟着今天的教程一步步来实现吧. 目标 设本地有一个demo.js ...

  2. Android入门第20天-Android里的ScrollView的使用

    介绍 ScrollView(滚动条),它有两种"滚动条": 竖直滚动条: 水平方向上的滚动条:HorizontalScrollView: 我们经常可以看到在手机里正在垂直加载一堆的 ...

  3. Android入门第19天-Android里的RatingBar的使用

    介绍 我们先来看一下什么叫RatingBar长什么样的. 我们很多时候订单评价给5星.打车评价都是用的这个RatingBar.本节我们学的这个RatingBar(星级评分条)其实在使用上是非常简单的, ...

  4. Android入门第31天-Android里的ViewFlipper翻转视图的使用

    介绍 本篇给大家带了的是ViewFlipper,它是Android自带的一个多页面管理控件,且可以自动播放! 和ViewPager不同,ViewPager是一页页的,而ViewFlipper则是一层层 ...

  5. Android入门第60天-MVVM中的Databinding与ListView结合使用

    开篇 还记得我们进入Listview.GridView都是以一个layout+adapter组合在一起来实现的是吧?那么还记得我们的Adapter的写法么? 在我们的Adapter里提供了一个bind ...

  6. Android入门第2天-Android Studio中新建项目

    本章要完成的目标 创建新工程: 设置Gradle镜像: 先讲一下Android Studio工具的使用常识 Android Studio是一款基于IDE的Android开发工具,为目前市面最标准And ...

  7. Android入门第1天-Android Studio的安装

    安装前的准备 在hosts文件中加入dl.google.com的ip映射 Android Studio在安装时大量的依赖库会从dl.google.com下载.在国内安装步骤卡死基本都是卡死在该步.那么 ...

  8. Android入门第30天-Android里的Toast的使用

    介绍 本篇带来的是: Android用于提示信息的一个控件--Toast(吐司)!Toast是一种很方便的消息提示框,会在 屏幕中显示一个消息提示框,没任何按钮,也不会获得焦点一段时间过后自动消失! ...

  9. Android入门第18天-Android里的SeekBar的使用

    SeekBar的介绍 SeekingBar是这么样的一个东西,它常常用在播放视频.音步时用来调节音量.模糊阀值等场景. SeekBar里怎么是通过:android:layout_weight=&quo ...

最新文章

  1. github README.md教程
  2. Ways to 优化JAVA程序设计和编码,提高JAVA性能
  3. 两篇文章说异常和中断之二
  4. 儿童学python编程入门用途-干货 | 看了此文,家长就知道为啥要让孩子学Python?...
  5. python开发好学吗-Python是不是很难学?
  6. 关于成为一名优秀的软件测试工程师
  7. EntityFramwork常见问题
  8. c语言简答程序源代码,C语言简答题答案
  9. asp.net 初步入门使用正则抓取网页信息
  10. 基于uFUN开发板的心率计(三)Qt上位机的实现
  11. Hadoop单机环境搭建整体流程
  12. 看完这篇,终于知道自己会不会 C# 泛型了!
  13. ORACLE自增字段创建方法
  14. 飞冰:Iceworks 自定义模板支持布局定制(v2.3.0 版本)
  15. Latex给表格加脚注
  16. VMware 磁碟機未備妥
  17. PMP|项目经理如何做好相关方管理?
  18. 寒假每日一题题解(1.29)摘花生(DP水题)
  19. 带你学会C++文字页面类项目——4.整体完善与项目1
  20. GPS接收机-从射频信号到定位解算

热门文章

  1. 传输指令ssh,sftp,scp
  2. 终于有人把工业数据采集讲明白了
  3. 10.27 动词未然形
  4. 给X系列手机设置安全区域
  5. H2教程系列(一) 简介与安装
  6. CMMI终于评估完了
  7. LSPG1000-激波管压力传感器测试系统
  8. asp毕业设计——基于asp+sqlserver的工厂设备管理系统设计与实现(毕业论文+程序源码)——工厂设备管理系统
  9. 初中学历可以自学matlab,过来人对初中生的忠告|初中三年,如何少走弯路?
  10. python文字转语音女声_python文字转语音