Android入门之IntentService的使用教程详解

目录
  • 开篇
  • IntentService
  • 课程目标
  • 代码核心设计
    • service注册
    • Service类-LongWaitingService
    • 主类-MainActivity.java
  • 运行效果

开篇

在前一篇中我们讲了bindService的使用。并且我们留下了一个念想,即在bindService取值时故意阻塞30秒,引起了一次ANR并引出了今天的章节-IntentService。

IntentService的生命周期中有一个非常好的方法-onHandleIntent方法,它是一个abstract方法,开发者在实现IntentService时可以覆盖它来处理“长事务”。

IntentService

Android开发者官网说过:

  • Service不是一个单独的进程,它和它的应用程序在同一个进程中
  • Service不是一个线程,这样就意味着我们应该避免在Service中进行耗时操作

于是乎,Android给我们提供了解决上述问题的替代品,就是下面要讲的IntentService; IntentService是继承与Service并处理异步请求的一个类,在IntentService中有 一个工作线程来处理耗时操作,请求的Intent记录会加入队列。

这么神奇?

我们来看演示,如何来验证这个IntentService里的onHandleIntent处理这种长事务。

课程目标

设Service里有一个字符型数组:

private String[] stdNames = new String[]{"小王", "小明", "小张"};

在Activity里输入数组下标后、等待30秒、然后把相对应的数组下标所对应的StudentName显示在Toast里,看看是不是会发生ANR。

该点击动作可以反复点击,因为每次点击后都会执行unbindService。

代码核心设计

IntentService没什么特殊的,它只是extends 自 IntentService,同时它拥有一个可以被覆盖的:onHandleIntent方法。

  • 我们这次使用CallBack模式来实现Service里长事务结束后回调activity里的handler实现数值传递;
  • 我们使用intent.putExtra来实现activity里的数值传递到service中去;

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">

    <application
        android: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.DemoRealIntentService"
        tools:targetApi="31">
        <service
            android:name=".LongWaitingService"
            android:exported="false"></service>

        <activity
            android:name=".MainActivity"
            android:exported="true">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>

            <meta-data
                android:name="android.app.lib_name"
                android:value="" />
        </activity>
    </application>

</manifest>

Service类-LongWaitingService

package org.mk.android.demorealintentservice;

import android.app.IntentService;
import android.content.Intent;
import android.content.Context;
import android.os.IBinder;
import android.util.Log;

public class LongWaitingService extends IntentService {
    private final String TAG = "LongWaitingService";
    private String[] stdNames = new String[]{"小王", "小明", "小张"};
    private Callback callback;
    private int stdNo;

    public class StudentBinder extends android.os.Binder {
        public LongWaitingService getService() {
            return LongWaitingService.this;
        }
    }

    public void setCallback(Callback callback) {
        this.callback = callback;
    }

    public static interface Callback {
        void onDataChange(String data);
    }

    public LongWaitingService() {
        super("LongWaitingService");
    }

    @Override
    public void onStart(Intent intent, int startId) {
        Log.i(TAG, ">>>>>>onStart");
    }
        @Override
    protected void onHandleIntent(Intent intent) {
        if (intent != null) {
            Log.i(TAG, ">>>>>>onHandleIntent");
            Log.i(TAG, ">>>>>>into a long waiting");
            new Thread() {
                public void run() {
                    try {
                        Thread.sleep(30000);
                        if (callback != null) {
                            String stdName = stdNames[stdNo];
                            callback.onDataChange(stdName);
                        }
                    } catch (Exception e) {
                    }
                }
            }.start();
        }
    }
    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        Log.i(TAG, ">>>>>>onBind方法被调用");
        this.stdNo = intent.getIntExtra("stdNo", -1);
        onHandleIntent(intent);
        return new StudentBinder();
    }
    //Service被关闭前回调
    @Override
    public void onDestroy() {
        super.onDestroy();
        Log.i(TAG, ">>>>>>onDestroyed方法被调用!");
    }
    @Override

    public boolean onUnbind(Intent intent) {

        Log.i(TAG,">>>>>>onUnbind");

        return false;

    }
}

主类-MainActivity.java

package org.mk.android.demorealintentservice;

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.Handler;
import android.os.IBinder;
import android.os.Message;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {
    private final String TAG = "LongWaitingService";
    private Button buttonGetValueFromBinder;
    private Button buttonClose;
    private Context ctx;
    private Intent intent;
    private LongWaitingService.StudentBinder stdBinder;
    private EditText etStdNo;
    Handler stdHandler = new StudentHandler();
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        buttonGetValueFromBinder = (Button) findViewById(R.id.buttonGetValueFromBinder);
        etStdNo = (EditText) findViewById(R.id.etStdNo);
        ctx = MainActivity.this;
        intent = new Intent(ctx, LongWaitingService.class);

        buttonGetValueFromBinder.setOnClickListener(new OnClickListener());
    }
    private ServiceConnection conn = new ServiceConnection() {

        //Activity与Service断开连接时回调该方法
        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.i(TAG, ">>>>>>Service DisConnected");
        }

        //Activity与Service连接成功时回调该方法
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.i(TAG, ">>>>>>Service Connected");
            stdBinder = (LongWaitingService.StudentBinder) service;
            LongWaitingService stdService = stdBinder.getService();
            stdService.setCallback(new LongWaitingService.Callback() {
                @Override
                public void onDataChange(String data) {
                    Message msg = new Message();
                    msg.obj = data;
                    stdHandler.sendMessage(msg);
                }
            });
        }
    };

    class OnClickListener implements View.OnClickListener {
        @Override
        public void onClick(View view) {
            Intent eIntent;
            switch (view.getId()) {
                case R.id.buttonGetValueFromBinder:
                    int stdNo = Integer.valueOf(etStdNo.getText().toString());
                    intent.putExtra("stdNo", stdNo);
                    bindService(intent, conn, Service.BIND_AUTO_CREATE);
                    break;
            }
        }
    }
    class StudentHandler extends Handler {
        @Override
        public void handleMessage(Message msg) {
            Log.i(TAG,">>>>>>Service的count" + "的值为:" + msg.obj.toString());
            Toast.makeText(getApplicationContext(), "Service的count" + "的值为:" + msg.obj.toString(), Toast.LENGTH_LONG).show();
            unbindService(conn);
        }
    }
}

运行效果

来看看运行效果吧

看,再也没有ANR了,结果成功通过CALL BACK回传Activity。

因此我们一般都会这么使用IntentService来实现一些资源异步加载、第三方API回调。

到此这篇关于Android入门之IntentService的使用教程详解的文章就介绍到这了,更多相关Android IntentService内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Android开发Intent跳转传递list集合实现示例

    目录 引言 传递list集合 传递ArrayList集合 传递实体类 传递String 引言 Android 两个activity之间普通传值 如:单个String ,int ... 就不多说了 参考文章 https://www.jb51.net/article/257178.htm 传递list集合 注意:list传值的实体类用序列化 先创建个实体类(等下测试用) import java.io.Serializable; //别忘了序列化 Serializable public class D

  • Android显式Intent与隐式Intent的使用详解

    目录 什么是Intent 一.显式Intent和隐式Intent 1.显式Intent (1)在Intent构造函数中指定 (2)调用意图对象的setClass方法指定 (3)调用意图对象的setComponent方法指定 2.隐式Intent 什么是Intent Intent是各个组件之间信息沟通的桥梁,它用于Android各组件之间的通信,主要完成下列工作: 标明本次通信请求从哪里来.到哪里去.要怎么走. 发起方携带本次通信需要的数据内容,接收方从收到的意图中解析数据. 发起方若想判断接收方

  • 关于Android Activity之间跳转问题(Intent)

    当一个Acitivity需要启动另一个Activity时,通过Intent来表达自己的意图,告知系统启动哪个Activity. private ImageButton i2; i2=findViewById(R.id.i2); i2.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View v) { Intent intent=new Intent(UserActivity.this,Us

  • Android Intent传递大量数据出现问题解决

    目录 正文 异常TransactionTooLargeException Intent携带信息的大小受Binder限制 为什么Binder要限制传输数据的大小 替代方案 正文 官方文档 https://developer.android.google.cn/guide/components/activities/parcelables-and-bundles 在sendBroadcast,startActivity时,我们会用到Intent. Intent可以携带一些数据,比如基本类型数据int

  • Android中Intent与Bundle的使用详解

    目录 Intent android中Intent和Bundle之间的关系 Intent Intent是Android程序中个组件之间进行交互的一种重要方式,它不仅可以指明当前组件想要执行的动作,可以指定你想要启动的活动,也可以在不同组件之间传递数据. 第一部分:通过intent来启动一个活动 Intent 分为显示和隐式 我们先来了解一下显示 其中一个常用的构造方法是: Intent intent=new Intent(registActivity.this,showActivity.class

  • Android入门之Handler的使用教程详解

    目录 简介 项目结构 代码 前端 后端 简介 我们在前面的Android教程中已经提到过这么一件事:Android在启动后会有一个主线程.它不允许任何子线程去改变主UI线程里的内容. 这么做是为了妨止,万一有一个带有大事务的线程导致了渲染组件时间过长最终导致Android UI出现“闪退”.“崩溃”的保护机制. 而实际我们在Android操作里的确是会有一些“耗时”的事情而采用异步线程,如:首页打开时调用第三方地图定位API.调用第三方银行API来显示你的余额.调用第三方社保显示你的当前社保缴纳

  • Android入门之使用SharedPreference存取信息详解

    目录 简介 SharedPreference原理 SharedPreference的存储 后端代码 SharedPreferenceHelper.java HomePage.java MainActivity.java 简介 上一篇我们介绍了在android里如何读写本地文件.我们有一种场景,类似网页的cookie,要把用户的一些储如上一次登录.使用的痕迹等信息保存下来以便于每次不需要做重复“填表单”的操作,当在这种场景下我们如果也使用本地文件读写的话显然是“太重”了.因此android提供了一

  • Android入门之Activity间互相传值详解

    目录 介绍 Android中的传值 课程目标 全代码 前端 后端代码 附.工程结构 介绍 今天的课程会比较好玩,我们在之前的Service篇章中看到了一种putExtras和getExtras来进行activity与service间的传值.而恰恰这种传值其实也是Android里的通用传值法.它同样可以适用在activity与activity间传值. Android中的传值 传单个值 传多个值 具体我们来结合例子来看吧 课程目标 正向传值到下一个activity上 反向传值到调用activity上

  • ABP(现代ASP.NET样板开发框架)系列之二、ABP入门教程详解

    ABP是"ASP.NET Boilerplate Project (ASP.NET样板项目)"的简称. ASP.NET Boilerplate是一个用最佳实践和流行技术开发现代WEB应用程序的新起点,它旨在成为一个通用的WEB应用程序框架和项目模板. ABP的官方网站:http://www.aspnetboilerplate.com ABP在Github上的开源项目:https://github.com/aspnetboilerplate ABP 的由来 "DRY--避免重复

  • Vuejs第一篇之入门教程详解(单向绑定、双向绑定、列表渲染、响应函数)

    什么是组件? 组件(Component)是 Vue.js 最强大的功能之一.组件可以扩展 HTML 元素,封装可重用的代码.在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功能.在有些情况下,组件也可以是原生 HTML 元素的形式,以 is 特性扩展. 接下来给大家介绍vuejs单向绑定.双向绑定.列表渲染.响应函数基础知识,具体详情如下所示: (一)单向绑定 <div id="app"> {{ message }} </div> <sc

  • BootStrop前端框架入门教程详解

    Bootstrap,来自 Twitter,是目前最受欢迎的前端框架.Bootstrap 是基于 HTML.CSS.JAVASCRIPT 的,它简洁灵活,使得 Web 开发更加快捷. bootstrap 的学习非常简单,并且它所提供的样式又非常精美.只要稍微简单的学习就可以制作出漂亮的页面. bootstrap中文网:http://v3.bootcss.com/ bootstrap提供了三种类型的下载: -----------------------------------------------

  • Node.js+Express配置入门教程详解

    Node.js是一个Javascript运行环境(runtime).实际上它是对Google V8引擎进行了封装.V8引 擎执行Javascript的速度非常快,性能非常好.Node.js对一些特殊用例进行了优化,提供了替代的API,使得V8在非浏览器环境下运行得更好.Node.js是一个基于Chrome JavaScript运行时建立的平台, 用于方便地搭建响应速度快.易于扩展的网络应用.Node.js 使用事件驱动, 非阻塞I/O 模型而得以轻量和高效,非常适合在分布式设备上运行的数据密集型

  • Android开发实现带有反弹效果仿IOS反弹scrollview教程详解

    首先给大家看一下我们今天这个最终实现的效果图: 这个是ios中的反弹效果.当然我们安卓中如果想要实现这种效果,感觉不会那么生硬,滚动到底部或者顶部的时候.当然 使用scrollview是无法实现的.所以我们需要新建一个view继承ScrollView package davidbouncescrollview.qq986945193.com.davidbouncescrollview; import android.annotation.SuppressLint; import android.

  • Android Studio开发环境搭建教程详解

    对于移动端这块,笔者之前一直都是进行iOS开发的,也从来没用过Java.但是因为进入了Google Android全国大学生移动互联网创新挑战赛(进入官网)的总决赛(笔者"西部计算机教育提升计划"的项目被直接推荐进入决赛),这个比赛要求一定要提交apk程序,所以我不得不赶紧学习一下Android开发了. 下面就对自己学习的过程做一个记录. 一.安装Android Studio 笔者用的计算机配置如下: Mac下安装Android Studio应该更简单一些,只需要下载一个Android

  • Android Studio 新建项目通过git上传到码云图文教程详解

    作为一个合格的开发人员,对项目进行管理自然必不可少.今天就给各位看客介绍一下如何用git将自己的AS项目上传到码云.    一  创建远程码云代码仓库 首先我们打开码云,注册一个账号,接着在左上角处点击加号新建一个远程的项目仓库用于待会AS项目的上传,具体如下: 按照上图所示,点击创建即生成远程代码仓库.     二 开始进行AS代码的上传 首先我们打开新建好的AS项目,点击AS顶部的如下: 接着点击项目名称如下将项目加入到本地的git仓库中 之后红色的字体会变成绿色. 接着点击顶部vcs上传代

随机推荐