Android基础入门教程——4.2.2 Service进阶

Android基础入门教程——4.2.2 Service进阶

标签(空格分隔): Android基础入门教程


本节引言

上节我们学习了Service的生命周期,以及两种启动Service的两种方法,

本节继续来深入了解Service中的IntentService,Service的使用实例:

前台服务与轮询的实现!


1.IntentService的使用

在上一节后我们已经知道了如何去定义和启动Service,但是如果我们直接把

耗时线程放到Service中的onStart()方法中,虽然可以这样做,但是很容易

会引起ANR异常(Application Not Responding),而Android的官方在介绍

Service有下面这样一段话:

直接翻译:

1.Service不是一个单独的进程,它和它的应用程序在同一个进程中

2.Service不是一个线程,这样就意味着我们应该避免在Service中进行耗时操作

于是乎,Android给我们提供了解决上述问题的替代品,就是下面要讲的IntentService

IntentService是继承与Service并处理异步请求的一个类,在IntentService中有

一个工作线程来处理耗时操作,请求的Intent记录会加入队列

工作流程:

客户端通过startService(Intent)来启动IntentService;

我们并不需要手动地区控制IntentService,当任务执行完后,IntentService会自动停止;

可以启动IntentService多次,每个耗时操作会以工作队列的方式在IntentService的

onHandleIntent回调方法中执行,并且每次只会执行一个工作线程,执行完一,再到二这样!

再接着是代码演示,网上大部分的代码都是比较Service与IntentService的,

定义足够长的休眠时间,演示Service的ANR异常,然后引出IntentService有多好!

这里就不演示Service了,网上的都是自定义Service,然后在onStart()方法

中Thread.sleep(20000)然后引发ANR异常,有兴趣的可以自己写代码试试,

这里的话只演示下IntentService的用法!

TestService3.java

public class TestService3 extends IntentService {
    private final String TAG = "hehe";
    //必须实现父类的构造方法
    public TestService3()
    {
        super("TestService3");
    }  

    //必须重写的核心方法
    @Override
    protected void onHandleIntent(Intent intent) {
        //Intent是从Activity发过来的,携带识别参数,根据参数不同执行不同的任务
        String action = intent.getExtras().getString("param");
        if(action.equals("s1"))Log.i(TAG,"启动service1");
        else if(action.equals("s2"))Log.i(TAG,"启动service2");
        else if(action.equals("s3"))Log.i(TAG,"启动service3");  

        //让服务休眠2秒
        try{
            Thread.sleep(2000);
        }catch(InterruptedException e){e.printStackTrace();}
    }  

    //重写其他方法,用于查看方法的调用顺序
    @Override
    public IBinder onBind(Intent intent) {
        Log.i(TAG,"onBind");
        return super.onBind(intent);
    }  

    @Override
    public void onCreate() {
        Log.i(TAG,"onCreate");
        super.onCreate();
    }  

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.i(TAG,"onStartCommand");
        return super.onStartCommand(intent, flags, startId);
    }  

    @Override
    public void setIntentRedelivery(boolean enabled) {
        super.setIntentRedelivery(enabled);
        Log.i(TAG,"setIntentRedelivery");
    }  

    @Override
    public void onDestroy() {
        Log.i(TAG,"onDestroy");
        super.onDestroy();
    }  

}  

AndroidManifest.xml注册下Service

<service android:name=".TestService3" android:exported="false">
    <intent-filter >
        <action android:name="com.test.intentservice"/>
    </intent-filter>
</service>  

在MainActivity启动三次服务:

public class MainActivity extends Activity {  

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);  

        Intent it1 = new Intent("com.test.intentservice");
        Bundle b1 = new Bundle();
        b1.putString("param", "s1");
        it1.putExtras(b1);  

        Intent it2 = new Intent("com.test.intentservice");
        Bundle b2 = new Bundle();
        b2.putString("param", "s2");
        it2.putExtras(b2);  

        Intent it3 = new Intent("com.test.intentservice");
        Bundle b3 = new Bundle();
        b3.putString("param", "s3");
        it3.putExtras(b3);  

        //接着启动多次IntentService,每次启动,都会新建一个工作线程
        //但始终只有一个IntentService实例
        startService(it1);
        startService(it2);
        startService(it3);
    }
}  

运行截图:

小结:

当一个后台的任务,需要分成几个子任务,然后按先后顺序执行,子任务

(简单的说就是异步操作),此时如果我们还是定义一个普通Service然后

在onStart方法中开辟线程,然后又要去控制线程,这样显得非常的繁琐;

此时应该自定义一个IntentService然后再onHandleIntent()方法中完成相关任务!


2.Activity与Service通信

我们前面的操作都是通过Activity启动和停止Service,假如我们启动的是一个下载

的后台Service,而我们想知道Service中下载任务的进度!那么这肯定是需要Service

与Activity进行通信的,而他们之间交流的媒介就是Service中的onBind()方法!

返回一个我们自定义的Binder对象!

基本流程如下:

1.自定义Service中,自定义一个Binder类,然后将需要暴露的方法都写到该类中!

2.Service类中,实例化这个自定义Binder类,然后重写onBind()方法,将这个Binder对象返回!

3.Activity类中实例化一个ServiceConnection对象,重写onServiceConnected()方法,然后

获取Binder对象,然后调用相关方法即可!


3.一个简单前台服务的实现

学到现在,我们都知道Service一般都是运行在后来的,但是Service的系统优先级

还是比较低的,当系统内存不足的时候,就有可能回收正在后台运行的Service,

对于这种情况我们可以使用前台服务,从而让Service稍微没那么容易被系统杀死,

当然还是有可能被杀死的…所谓的前台服务就是状态栏显示的Notification!

实现起来也很简单,最近做的项目刚好用到这个前台服务,就把核心的代码抠出来

分享下:

在自定义的Service类中,重写onCreate(),然后根据自己的需求定制Notification;

定制完毕后,调用startForeground(1,notification对象)即可!

核心代码如下:

public void onCreate()
  {
    super.onCreate();
    Notification.Builder localBuilder = new Notification.Builder(this);
    localBuilder.setContentIntent(PendingIntent.getActivity(this, 0, new Intent(this, MainActivity.class), 0));
    localBuilder.setAutoCancel(false);
    localBuilder.setSmallIcon(R.mipmap.ic_cow_icon);
    localBuilder.setTicker("Foreground Service Start");
    localBuilder.setContentTitle("Socket服务端");
    localBuilder.setContentText("正在运行...");
    startForeground(1, localBuilder.getNotification());
  }

运行效果截图:


4.简单定时后台线程的实现

除了上述的前台服务外,实际开发中Service还有一种常见的用法,就是执行定时任务,

比如轮询,就是每间隔一段时间就请求一次服务器,确认客户端状态或者进行信息更新

等!而Android中给我们提供的定时方式有两种使用Timer类与Alarm机制!

前者不适合于需要长期在后台运行的定时任务,CPU一旦休眠,Timer中的定时任务

就无法运行;Alarm则不存在这种情况,他具有唤醒CPU的功能,另外,也要区分CPU

唤醒与屏幕唤醒!

使用流程:

  • Step 1:获得Service:

    AlarmManager manager = (AlarmManager) getSystemService(ALARM_SERVICE);

  • Step 2:通过set方法设置定时任务

    int anHour = 2 * 1000;

    long triggerAtTime = SystemClock.elapsedRealtime() + anHour;

    manager.set(AlarmManager.RTC_WAKEUP,triggerAtTime,pendingIntent);

  • Step 3:定义一个Service

    在onStartCommand中开辟一条事务线程,用于处理一些定时逻辑

  • Step 4:定义一个Broadcast(广播),用于启动Service

    最后别忘了,在AndroidManifest.xml中对这Service与Boradcast进行注册!

参数详解:

set(int type,long startTime,PendingIntent pi)

①type:

有五个可选值:

AlarmManager.ELAPSED_REALTIME:

闹钟在手机睡眠状态下不可用,该状态下闹钟使用相对时间(相对于系统启动开始),状态值为3;

AlarmManager.ELAPSED_REALTIME_WAKEUP

闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟也使用相对时间,状态值为2;

AlarmManager.RTC

闹钟在睡眠状态下不可用,该状态下闹钟使用绝对时间,即当前系统时间,状态值为1;

AlarmManager.RTC_WAKEUP

表示闹钟在睡眠状态下会唤醒系统并执行提示功能,该状态下闹钟使用绝对时间,状态值为0;

AlarmManager.POWER_OFF_WAKEUP

表示闹钟在手机关机状态下也能正常进行提示功能,所以是5个状态中用的最多的状态之一,

该状态下闹钟也是用绝对时间,状态值为4;不过本状态好像受SDK版本影响,某些版本并不支持;

PS:第一个参数决定第二个参数的类型,如果是REALTIME的话就用:

SystemClock.elapsedRealtime( )方法可以获得系统开机到现在经历的毫秒数

如果是RTC的就用:System.currentTimeMillis()可获得从1970.1.1 0点到

现在做经历的毫秒数

②startTime:

闹钟的第一次执行时间,以毫秒为单位,可以自定义时间,不过一般使用当前时间。

需要注意的是,本属性与第一个属性(type)密切相关,如果第一个参数对应的闹钟

使用的是相对时间(ELAPSED_REALTIMEELAPSED_REALTIME_WAKEUP),那么本属

性就得使用相对时间(相对于系统启动时间来说),比如当前时间就表示为:

SystemClock.elapsedRealtime();如果第一个参数对应的闹钟使用的是绝对时间 (RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那么本属性就得使用绝对时间,

比如当前时间就表示为:System.currentTimeMillis()。

③PendingIntent:

绑定了闹钟的执行动作,比如发送一个广播、给出提示等等。PendingIntent

是Intent的封装类。需要注意的是,如果是通过启动服务来实现闹钟提示的话,

PendingIntent对象的获取就应该采用Pending.getService

(Context c,int i,Intent intent,int j)方法;如果是通过广播来实现闹钟提示的话,

PendingIntent对象的获取就应该采用 PendingIntent.getBroadcast

(Context c,int i,Intent intent,int j)方法;

如果是采用Activity的方式来实现闹钟提示的话,PendingIntent对象的获取

就应该采用 PendingIntent.getActivity(Context c,int i,Intent intent,int j)

方法。如果这三种方法错用了的话,虽然不会报错,但是看不到闹钟提示效果。

另外:

从4.4版本后(API 19),Alarm任务的触发时间可能变得不准确,有可能会延时,是系统

对于耗电性的优化,如果需要准确无误可以调用setExtra()方法~

核心代码:

LongRunningService.java

public class LongRunningService extends Service {
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        //这里开辟一条线程,用来执行具体的逻辑操作:
        new Thread(new Runnable() {
            @Override
            public void run() {
                Log.d("BackService", new Date().toString());
            }
        }).start();
        AlarmManager manager = (AlarmManager) getSystemService(ALARM_SERVICE);
        //这里是定时的,这里设置的是每隔两秒打印一次时间=-=,自己改
        int anHour = 2 * 1000;
        long triggerAtTime = SystemClock.elapsedRealtime() + anHour;
        Intent i = new Intent(this,AlarmReceiver.class);
        PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0);
        manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, triggerAtTime, pi);
        return super.onStartCommand(intent, flags, startId);
    }
}

AlarmReceiver.java

public class AlarmReceiver extends BroadcastReceiver {

    @Override
    public void onReceive(Context context, Intent intent) {
        Intent i = new Intent(context,LongRunningService.class);
        context.startService(i);
    }
}

本节小结:

本节我们继续对Service进行更深入的学习,IntentService以及Service

在实际开发中的两个常用的案例:前台Service的实现,以及Service后台

Service的实现!下一节中我们会继续研究Service的AIDL,跨进程通信,

敬请期待~



参考文献:

《第一行代码 Android》—— 郭霖:很好的一本Android入门书!

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-11-17 12:30:47

Android基础入门教程——4.2.2 Service进阶的相关文章

Android基础入门教程——4.2.3 Service精通

Android基础入门教程--4.2.3 Service精通 标签(空格分隔): Android基础入门教程 本节引言: 本节,我们继续来研究Service(服务)组件,本节将会学习下Android中的AIDL跨进程通信的一些 概念,并不深入到源码层次,暂时知道是什么,会用即可!开始本节内容~ 本节对应官方文档:Binder 1.Binder机制初涉 1)IBinder和Binder是什么鬼? 我们来看看官方文档怎么说: 中文翻译: IBinder是远程对象的基本接口,是饿了高性能而设计的轻量级

Android基础入门教程——4.2.1 Service初涉

Android基础入门教程--4.2.1 Service初涉 标签(空格分隔): Android基础入门教程 本节引言 好的,我们在前三节中对Android中的Activity进行了研究学习,相信大家获益良多吧! 本节开始我们继续来学习Android中的第二个组件:Service(服务), 好,废话不多说,开始本节内容! 1.线程的相关概念 在开始学习Service之前我们先来了解下线程的一些概念! 1)相关概念: 程序:为了完成特定任务,用某种语言编写的一组指令集合(一组静态代码) 进程:运行

2015年最新Android基础入门教程目录(完结版)

2015年最新Android基础入门教程目录(完结版) 标签(空格分隔): Android基础入门教程 前言: 关于<2015年最新Android基础入门教程目录>终于在今天落下了帷幕,全套教程 共148节已编写完毕,附上目录,关于教程的由来,笔者的情况和自学心得,资源分享 以及一些疑问等可戳:<2015最新Android基础入门教程>完结散花~ 下面是本系列教程的完整目录: 第一章:环境搭建与开发相关(已完结 10/10) Android基础入门教程--1.1 背景相关与系统架构

2015年最新Android基础入门教程目录(临时版)

2015年最新Android基础入门教程目录(临时版) 标签(空格分隔): Android基础入门教程 前言: 嗯,昨晚又给人盗号了,博客上被发表了十几篇黄贴-然后目录给管理误删了,再发一次 后来协商后发现实被设密保问题了,建议各位用csdn的朋友密保自己设置一波~ 密保问题已修改回来了,应该不会再被盗号了-人怕出名猪怕壮哈~下次如果发现博客被封 告知下小猪,如何很急的话可以先到w3c鸟巢菜鸟教程上看Android基础入门教程 经过站长FK进行排版的,可能阅读体验会比csdn好很多!内容基本是同

Android基础入门教程——10.10 传感器专题(1)——相关介绍

Android基础入门教程--10.10 传感器专题(1)--相关介绍 标签(空格分隔): Android基础入门教程 1.传感器相关介绍: 说到传感器,相信大家都不会陌生吧,比如微信的摇一摇就用到了加速度传感器: 传感器的定义:一种物理设备或者生物器官,能够探测.感受外界的信号,物理条件(如光,热, 适度)或化学组成(如烟雾),并将探知的信息传递给其他的设备或者器官! 传感器的种类:可以从不同的角度对传感器进行划分,转换原理(传感器工作的基本物理或化学 效应):用途:输出信号以及制作材料和工艺

Android基础入门教程——10.4 Vibrator(振动器)

Android基础入门教程--10.4 Vibrator(振动器) 标签(空格分隔): Android基础入门教程 本节引言: 本节我们介绍的是Vibrator(振动器),是手机自带的振动器,别去百度直接搜针振动器,因为 你的搜索结果可能是如图所示的神秘的道具,或者其他神秘道具: 嗯,说回本节介绍的Vibrator,其实就是Android给我们提供的用于机身震动的一个服务! 比如前面我们的Notification中可以设置震动,当收到推送消息的时候我们可以设置震动 提醒,游戏必备,比如"打飞机&

Android基础入门教程——1.2 开发环境搭建

Android基础入门教程--1.2 开发环境搭建 现在主流的Android开发环境有: ①Eclipse + ADT + SDK ②Android Studio + SDK ③IntelliJ IDEA + SDK 现在国内大部分开发人员还是使用的Eclipse,而谷歌宣布不再更新ADT后,并且官网也去掉了集成Android开发环境的Eclipse下载链接,各种现象都表示开发者最后都终将过渡到Android Studio,当然这段过渡时间会很长,但如果你是刚学Android的话建议直接冲And

Android基础入门教程——10.9 WallpaperManager(壁纸管理器)

Android基础入门教程--10.9 WallpaperManager(壁纸管理器) 标签(空格分隔): Android基础入门教程 本节引言: 本节给大家带来的是WallpaperManager(壁纸管理器),如其名,就是手机壁纸相关的 一个API,在本节中我们会描述下WallpaperManager的基本用法,调用系统自带的 壁纸选择功能,将Activity的背景设置为壁纸背景,以及写一个定时换壁纸的例子~ 好了,不BB,开始本节内容~ 官方API文档:WallpaperManager 1

Android基础入门教程——10.1 TelephonyManager(电话管理器)

Android基础入门教程--10.1 TelephonyManager(电话管理器) 标签(空格分隔): Android基础入门教程 本节引言: 本章节是Android基础入门教程的最后一章,主要讲解是一些零零散散的一些知识点,以及一些遗漏 知识点的补充,这些零散的知识点包括,各种系统服务的使用,比如本节的电话管理器,短信管理器, 振动器,闹钟,壁纸等等,还有传感器之类的东西!乱七八糟什么都有哈!好的,本节我们要学习的 是TelephonyManager,见名知义:用于管理手机通话状态,获取电